From 821a610bbf7d35d05ae6e3310912e1f9c58ec883 Mon Sep 17 00:00:00 2001 From: Mike Taylor Date: Wed, 31 Jan 2024 18:04:50 +0000 Subject: [PATCH] First steps towards validating XSLT --- src/settings/ScriptOK.js | 41 ++++++++++++++++++++++++++++++++++++++++ src/settings/StepForm.js | 11 +++++++++++ 2 files changed, 52 insertions(+) create mode 100644 src/settings/ScriptOK.js diff --git a/src/settings/ScriptOK.js b/src/settings/ScriptOK.js new file mode 100644 index 0000000..c1ab22a --- /dev/null +++ b/src/settings/ScriptOK.js @@ -0,0 +1,41 @@ +import React from 'react'; +import PropTypes from 'prop-types'; + +const ScriptOK = ({ xslText }) => { + if (xslText === '') { + return null; + } + + const parser = new DOMParser(); + const xsltProcessor = new XSLTProcessor(); + + console.log('xslText =', xslText); + const xslStylesheet = parser.parseFromString(xslText, 'application/xml'); + const errorNode = xslStylesheet.querySelector('parsererror'); + if (errorNode) { + let errorMessage = errorNode.textContent; + console.log('untrimmed =', errorMessage); + errorMessage = errorMessage.trim(); + console.log('trimmed =', errorMessage); + // This is often of the form: + // XML Parsing Error: syntax error Location: http://localhost:3000/settings/ha/step/10010?layer=edit Line Number 1, Column 1 + errorMessage = errorMessage.replace(/Location: https?:[^ ]+/, ''); + return 'Bad XML: ' + errorMessage; + } + + console.log('xslStylesheet =', xslStylesheet); + try { + xsltProcessor.importStylesheet(xslStylesheet); + } catch (e) { + // No value to the exception: it's effectively a boolean (thrown or not) + // The XML was not valid XSLT + return 'Bad XSLT'; + } + return 'Good XSLT'; +}; + +ScriptOK.propTypes = { + xslText: PropTypes.string, +}; + +export default ScriptOK; diff --git a/src/settings/StepForm.js b/src/settings/StepForm.js index 14c3ad8..0ef5e3d 100644 --- a/src/settings/StepForm.js +++ b/src/settings/StepForm.js @@ -9,6 +9,7 @@ import { isEqual } from 'lodash'; import setFieldData from 'final-form-set-field-data'; // XXX do we need this? import { RCF, CF } from '../components/CF'; import renderPaneFooter from './renderPaneFooter'; +import ScriptOK from './ScriptOK'; function validate(values) { @@ -33,6 +34,15 @@ const StepForm = (props) => { const { form, handleSubmit, onCancel, pristine, submitting } = props; const intl = useIntl(); + /* + console.log('xform = ', form); + console.log('getRegisteredFields =', form.getRegisteredFields()); + const getFieldState = form.getFieldState + console.log('getFieldState = ', getFieldState); + const scriptField = getFieldState('script'); + console.log('scriptField = ', form.scriptField); + */ + const noValue = { value: '', label: intl.formatMessage({ id: 'ui-harvester-admin.selectValue' }), @@ -65,6 +75,7 @@ const StepForm = (props) => { +