Skip to content

Commit

Permalink
feat: add custom xsd
Browse files Browse the repository at this point in the history
  • Loading branch information
lekotros committed Aug 7, 2023
1 parent dcfc44b commit c23f6d4
Show file tree
Hide file tree
Showing 12 changed files with 410 additions and 62 deletions.
19 changes: 19 additions & 0 deletions app/api/client.ts
Original file line number Diff line number Diff line change
Expand Up @@ -104,6 +104,25 @@ class ApiClient {
})
}

async xsdUpload (id: string, file: File, onProgress?: (p: any) => void): Promise<any> {
const { checksum } = await calculateChecksum(file)
const data = new FormData()

data.append('file', file)

return await axios({
method: 'post',
url: this.withUrl(`sessions/${id}/upload/xsd`),
data,
params: { checksum },
onUploadProgress: (p) => {
if (onProgress !== undefined) {
onProgress(p)
}
}
})
}

async validate (id: string): Promise<any> {
return await axios({
method: 'get',
Expand Down
11 changes: 11 additions & 0 deletions app/api/types.ts
Original file line number Diff line number Diff line change
Expand Up @@ -23,13 +23,24 @@ export interface ScriptConfigOption {
options?: any[]
}

export interface XSDUploadFile {
id: string
name: string
}

export interface XSDUpload {
name: string
files?: XSDUploadFile[]
}

export interface Session {
id: string
name: string
ref: string
created: number
stopped: number
files: string[]
xsdFiles?: XSDUpload[]
status: string
results: any[]
profile?: Profile
Expand Down
172 changes: 138 additions & 34 deletions app/components/CustomConfiguration.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -17,18 +17,22 @@ import {
Stack,
Typography,
OutlinedInput,
DialogActions
DialogActions,
Card
} from '@mui/material'
import { grey } from '@mui/material/colors'
import TuneOutlineIcon from '@mui/icons-material/TuneOutlined'
import ChevronRightIcon from '@mui/icons-material/ChevronRight'
import React from 'react'
import type { Profile, Script } from '../api/types'
import FileUpload, { type FileList } from './FileUpload'
import type { Profile, Script, Session, XSDUploadFile } from '../api/types'
import scriptData from '../public/scripts.json'
import useApiClient from '../hooks/useApiClient'

const scriptOptions = scriptData.filter(v => v.name !== 'xsd')

export interface CustomConfigurationProps {
session: Session | null
onNext: (profile: Profile) => void
disabled?: boolean
}
Expand Down Expand Up @@ -129,10 +133,18 @@ const ScriptRow = ({
)
}

const CustomConfiguration = (props: CustomConfigurationProps): JSX.Element => {
const CustomConfiguration = ({
session,
disabled,
onNext
}: CustomConfigurationProps): JSX.Element => {
const [schema, setSchema] = React.useState<string>('[email protected]')
const [schemaEntry, setSchemaEntry] = React.useState<string>('')
const [scripts, setScripts] = React.useState<string[]>(scriptOptions.map(v => v.name))
const [scriptOpts, setScriptOpts] = React.useState<Record<string, Record<string, any>>>({})
const [fileList, setFileList] = React.useState<Record<string, unknown>>({})
const [schemaFiles, setSchemaFiles] = React.useState<XSDUploadFile[]>([])
const apiClient = useApiClient()

const handleSelectSchema = (event: SelectChangeEvent): void => {
if (event.target?.name === '') {
Expand All @@ -142,6 +154,14 @@ const CustomConfiguration = (props: CustomConfigurationProps): JSX.Element => {
setSchema(event.target.value)
}

const handleSelectSchemaEntry = (event: SelectChangeEvent): void => {
if (event.target?.name === '') {
return
}

setSchemaEntry(event.target.value)
}

const handleScriptToggle = (v: Script): void => {
const i = scripts.indexOf(v.name)
const s = [...scripts]
Expand All @@ -163,16 +183,18 @@ const CustomConfiguration = (props: CustomConfigurationProps): JSX.Element => {
}

const handleNextClick = (): void => {
if (props.disabled === true) {
if (disabled === true) {
return
}

const xsdScript = {
...scriptData.find(v => v.name === 'xsd'),
config: { schema }
config: schema === 'custom'
? { schema: 'custom', entry: schemaEntry }
: { schema }
}

props.onNext({
onNext({
name: 'custom',
description: 'Custom configuration',
longDescription: '',
Expand All @@ -189,36 +211,118 @@ const CustomConfiguration = (props: CustomConfigurationProps): JSX.Element => {
})
}

React.useEffect(() => {
if (session == null) {
return
}
if (session.xsdFiles != null) {
const { xsdFiles } = session

setFileList(xsdFiles.reduce((o: Record<string, any>, { name }) => {
o[name] = {
name,
status: 'uploaded',
progress: 100
}
return o
}, {}) ?? {})
setSchemaFiles(xsdFiles.reduce((o: XSDUploadFile[], v) => {
if (v.files != null) {
o.push(...v.files)
}
return o
}, []) ?? [])
}
if (session.profile != null) {
const { profile } = session
const xsdScript = profile.scripts?.find(v => v.name === 'xsd')

setSchema(xsdScript?.config?.schema)
setSchemaEntry(xsdScript?.config?.entry)
}
}, [session, setFileList])

return (
<Stack spacing={4}>
<Stack spacing={2}>
<Typography variant="h5">Profile</Typography>
<Typography><b>1.</b> Begin by selecting which profile to use for validation</Typography>
<Typography>
<ul style={{ marginTop: 0 }}>
<li>NeTEx - The full NeTEx schema (<a href="https://github.com/NeTEx-CEN/NeTEx/tree/12848763e6a9340b703de048368f2dd518ac3e27" target="_blank" rel="noreferrer">more info</a>)</li>
<li>NeTEx Fast - NeTEx schema without constraint (<a href="https://github.com/NeTEx-CEN/NeTEx/tree/12848763e6a9340b703de048368f2dd518ac3e27" target="_blank" rel="noreferrer">more info</a>)</li>
<li>EPIP - NeTEx European Passenger Information Profile (<a href="https://data4pt.org/NeTEx/GraphicKit/Documention_of_XSD_for_EPIP.html" target="_blank" rel="noreferrer">more info</a>)</li>
<li>EPIP Light - NeTEx European Passenger Information Profile</li>
</ul>
</Typography>
<FormControl>
<InputLabel id="netex-schema-label">Schema</InputLabel>
<Select
labelId="netex-schema-label"
name="netex-schema"
value={schema}
onChange={handleSelectSchema}
>
<MenuItem key="netex" value="[email protected]">NeTEx (v1.2)</MenuItem>
<MenuItem key="netex-light" value="[email protected]">NeTEx Fast (v1.2)</MenuItem>
<MenuItem key="epip" value="[email protected]">EPIP (v1.1.1)</MenuItem>
<MenuItem key="epip-light" value="[email protected]">EPIP Fast (v1.1.1)</MenuItem>
</Select>
</FormControl>
<Stack spacing={8}>
<Stack spacing={4}>
<Stack spacing={2}>
<Typography variant="h4">Profile</Typography>
<Typography><b>1.</b> Begin by selecting which profile to use for validation</Typography>
<Typography>
<ul style={{ marginTop: 0 }}>
<li>NeTEx - The full NeTEx schema (<a href="https://github.com/NeTEx-CEN/NeTEx/tree/12848763e6a9340b703de048368f2dd518ac3e27" target="_blank" rel="noreferrer">more info</a>)</li>
<li>NeTEx Fast - NeTEx schema without constraint (<a href="https://github.com/NeTEx-CEN/NeTEx/tree/12848763e6a9340b703de048368f2dd518ac3e27" target="_blank" rel="noreferrer">more info</a>)</li>
<li>EPIP - NeTEx European Passenger Information Profile (<a href="https://data4pt.org/NeTEx/GraphicKit/Documention_of_XSD_for_EPIP.html" target="_blank" rel="noreferrer">more info</a>)</li>
<li>EPIP Light - NeTEx European Passenger Information Profile</li>
</ul>
</Typography>
<FormControl>
<InputLabel id="netex-schema-label">Profile</InputLabel>
<Select
labelId="netex-schema-label"
name="netex-schema"
value={schema}
onChange={handleSelectSchema}
>
<MenuItem key="netex" value="[email protected]">NeTEx (v1.2)</MenuItem>
<MenuItem key="netex-light" value="[email protected]">NeTEx Fast (v1.2)</MenuItem>
<MenuItem key="epip" value="[email protected]">EPIP (v1.1.1)</MenuItem>
<MenuItem key="epip-light" value="[email protected]">EPIP Fast (v1.1.1)</MenuItem>
<MenuItem key="custom" value="custom">Custom</MenuItem>
</Select>
</FormControl>
</Stack>
{schema === 'custom' && (
<Card style={{ padding: 16 }}>
<Stack spacing={3}>
<Stack spacing={1}>
<Typography variant="h5">Upload custom profile</Typography>
<Typography gutterBottom>Select which file to use as profile &apos;Select file(s)&apos;</Typography>
</Stack>
<Stack alignItems="center" spacing={2}>
<FileUpload
values={fileList}
disabled={false}
supportedFormats={['xml', 'zip']}
onUpload={async (file: any, cb: any) => {
await apiClient.xsdUpload(session?.id ?? '', file, cb)
.then(res => {
setSchemaFiles(res.data?.xsdFiles?.reduce((o: XSDUploadFile[], v: any) => {
if (v.files != null) {
o.push(...v.files)
}
return o
}, []) ?? [])
})
}}
onChange={(fileList: FileList) => {
setFileList({ ...fileList })
}}
onError={({ errorMessage }: { errorMessage: string }) => {
return `Error caught uploading file, message: ${errorMessage}`
}}
/>
</Stack>
<FormControl>
<InputLabel id="schema-entry">Main entry point</InputLabel>
<Select
labelId="schema-entry-label"
name="schema-entry"
value={schemaEntry}
onChange={handleSelectSchemaEntry}
disabled={Object.values(fileList).length === 0}
>
{schemaFiles.map(({ id, name }) => (
<MenuItem key={id} value={id}>{name}</MenuItem>
))}
</Select>
</FormControl>
</Stack>
</Card>
)}
</Stack>
<Stack spacing={2}>
<Typography variant="h5">Rules</Typography>
<Typography variant="h4">Rules</Typography>
<Typography><b>2.</b> In addition to the schema validation, we have also included a few optional rules that validate the consistency of the documents</Typography>
<List
sx={{
Expand All @@ -242,7 +346,7 @@ const CustomConfiguration = (props: CustomConfigurationProps): JSX.Element => {
</Stack>
<Stack alignItems="center">
<Button
disabled={props.disabled}
disabled={(disabled ?? false) || (schema === 'custom' && schemaEntry === '')}
variant="contained"
endIcon={<ChevronRightIcon />}
onClick={handleNextClick}
Expand Down
12 changes: 10 additions & 2 deletions app/components/FileUpload.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -61,13 +61,21 @@ export type FileList = Record<string, any>
export interface FileUploadProps {
values: FileList
disabled?: boolean
supportedFormats?: string[]
onUpload: (file: any, cb: (p: any) => void) => Promise<any>
onChange: (fileList: FileList, valid: boolean) => void
onError: (file: any) => JSX.Element | string
}

const FileUpload = (props: FileUploadProps): JSX.Element => {
const { values, disabled, onUpload, onChange, onError } = props
const {
values,
disabled,
supportedFormats = ['xml', 'zip', 'gzip', 'bzip', 'tar'],
onUpload,
onChange,
onError
} = props

const updateFileContext = (fileContext: any): void => {
values[fileContext.name] = fileContext
Expand Down Expand Up @@ -175,7 +183,7 @@ const FileUpload = (props: FileUploadProps): JSX.Element => {
background: '#f3f3f3'
}}
>
<Typography>Supported archive/compression formats are zip, gzip, bzip and tar</Typography>
<Typography>Supported formats are {supportedFormats.join(', ')}</Typography>
</Box>
)}
</>
Expand Down
4 changes: 3 additions & 1 deletion app/components/ValidationResult.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -46,6 +46,7 @@ const ValidationResult = (props: ValidationResultProps): JSX.Element => {
const apiClient = useApiClient()
const [anchorEl, setAnchorEl] = React.useState<null | HTMLElement>(null)
const open = Boolean(anchorEl)
const isCustom = session.profile?.scripts?.find(s => s.name === 'xsd')?.config?.schema === 'custom'

const handleValidateAnother = (): void => {
if (props.onValidateAnother !== undefined) {
Expand Down Expand Up @@ -205,8 +206,9 @@ const ValidationResult = (props: ValidationResultProps): JSX.Element => {
<Button
variant="contained"
onClick={handleValidateAnother}
disabled={isCustom}
>
Validate with this configuration
Validate with this configuration{isCustom && (<><br/>(unsupported when using custom xsd)</>)}
</Button>
</Grid>
</Grid>
Expand Down
4 changes: 2 additions & 2 deletions app/pages/jobs/[id]/custom.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -84,8 +84,8 @@ const Custom: NextPage = () => {
<Stack spacing={4}>
<ValidationStepper step={0} onBack={router.back} />
<JobStatus session={session} onValidate={handleNewValidationClick} />
<Typography variant="h4">Custom configuration</Typography>
<CustomConfiguration onNext={handleNext} disabled={disabled} />
<Typography variant="h3">Custom configuration</Typography>
<CustomConfiguration onNext={handleNext} disabled={disabled} session={session} />
</Stack>

<FullscreenLoader open={loading} />
Expand Down
17 changes: 15 additions & 2 deletions app/styles/theme.ts
Original file line number Diff line number Diff line change
Expand Up @@ -211,7 +211,19 @@ theme.components.MuiPaper = {
styleOverrides: {
root: {
borderRadius: 0,
boxShadow: 'none'
boxShadow: 'none',
background: 'white'
}
}
}

theme.components.MuiCard = {
styleOverrides: {
root: {
borderStyle: 'solid',
borderRadius: 8,
borderWidth: 1,
borderColor: 'rgba(0, 0, 0, 0.12)'
}
}
}
Expand Down Expand Up @@ -315,6 +327,7 @@ theme.components.MuiListItemText = {
theme.components.MuiOutlinedInput = {
styleOverrides: {
root: {
background: 'white'
}
}
}
Expand All @@ -334,7 +347,7 @@ theme.components.MuiLink = {
styleOverrides: {
root: {
color: 'inherit',
fontWeight: 500,
fontWeight: 500
}
}
}
Expand Down
Loading

0 comments on commit c23f6d4

Please sign in to comment.