datajunction-ui 0.0.1-rc.18 → 0.0.1-rc.19

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dj-logo.svg ADDED
@@ -0,0 +1,10 @@
1
+ <svg width="83" height="83" viewBox="0 0 83 83" fill="none" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink">
2
+ <rect width="83" height="83" fill="url(#pattern0)"/>
3
+ <rect width="83" height="83" stroke="black"/>
4
+ <defs>
5
+ <pattern id="pattern0" patternContentUnits="objectBoundingBox" width="1" height="1">
6
+ <use xlink:href="#image0_1_2" transform="scale(0.00364964)"/>
7
+ </pattern>
8
+ <image id="image0_1_2" width="274" height="274" xlink:href=""/>
9
+ </defs>
10
+ </svg>
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "datajunction-ui",
3
- "version": "0.0.1-rc.18",
3
+ "version": "0.0.1-rc.19",
4
4
  "description": "DataJunction Metrics Platform UI",
5
5
  "module": "src/index.tsx",
6
6
  "repository": {
@@ -23,9 +23,10 @@
23
23
  "@babel/core": "^7.18.2",
24
24
  "@babel/preset-env": "^7.18.2",
25
25
  "@babel/preset-react": "7.18.6",
26
+ "@codemirror/lang-sql": "^6.4.0",
26
27
  "@reduxjs/toolkit": "1.8.5",
27
- "@testing-library/jest-dom": "5.16.5",
28
- "@testing-library/react": "13.4.0",
28
+ "@testing-library/jest-dom": "6.1.2",
29
+ "@testing-library/react": "14.0.0",
29
30
  "@types/fontfaceobserver": "^2.1.0",
30
31
  "@types/jest": "^27.5.2",
31
32
  "@types/node": "^14.18.27",
@@ -39,8 +40,12 @@
39
40
  "@types/testing-library__jest-dom": "^5.14.5",
40
41
  "@types/webpack": "^5.28.0",
41
42
  "@types/webpack-env": "^1.18.0",
43
+ "@uiw/codemirror-extensions-basic-setup": "4.21.12",
44
+ "@uiw/codemirror-extensions-langs": "4.21.12",
45
+ "@uiw/react-codemirror": "4.21.12",
42
46
  "babel-loader": "9.1.2",
43
47
  "chalk": "4.1.2",
48
+ "codemirror": "^6.0.0",
44
49
  "cronstrue": "2.27.0",
45
50
  "cross-env": "7.0.3",
46
51
  "css-loader": "6.8.1",
@@ -142,6 +147,9 @@
142
147
  ]
143
148
  },
144
149
  "jest": {
150
+ "transformIgnorePatterns": [
151
+ "!node_modules/"
152
+ ],
145
153
  "collectCoverageFrom": [
146
154
  "src/**/*.{js,jsx,ts,tsx}",
147
155
  "!src/**/*/*.d.ts",
@@ -159,13 +167,21 @@
159
167
  }
160
168
  }
161
169
  },
170
+ "resolutions": {
171
+ "@codemirror/state": "6.2.0",
172
+ "@codemirror/view": "6.2.0",
173
+ "@lezer/common": "^1.0.0"
174
+ },
162
175
  "devDependencies": {
163
176
  "@babel/plugin-proposal-class-properties": "7.18.6",
177
+ "@babel/plugin-proposal-private-property-in-object": "7.21.11",
178
+ "@testing-library/user-event": "14.4.3",
164
179
  "eslint-config-prettier": "8.8.0",
165
180
  "eslint-plugin-prettier": "4.2.1",
166
181
  "eslint-plugin-react-hooks": "4.6.0",
167
182
  "html-webpack-plugin": "5.5.1",
168
183
  "jest": "^29.5.0",
184
+ "jest-fetch-mock": "3.0.3",
169
185
  "mini-css-extract-plugin": "2.7.6"
170
186
  }
171
187
  }
@@ -0,0 +1,32 @@
1
+ const AlertIcon = props => (
2
+ <svg
3
+ width="2em"
4
+ height="2em"
5
+ viewBox="0 0 24 24"
6
+ version="1.1"
7
+ xmlns="http://www.w3.org/2000/svg"
8
+ >
9
+ <title>alert_fill</title>
10
+ <g id="page-1" stroke="none" strokeWidth="1" fill="none" fillRule="evenodd">
11
+ <g
12
+ id="System"
13
+ transform="translate(-48.000000, -48.000000)"
14
+ fillRule="nonzero"
15
+ >
16
+ <g id="alert_fill" transform="translate(48.000000, 48.000000)">
17
+ <path
18
+ d="M24,0 L24,24 L0,24 L0,0 L24,0 Z M12.5934901,23.257841 L12.5819402,23.2595131 L12.5108777,23.2950439 L12.4918791,23.2987469 L12.4918791,23.2987469 L12.4767152,23.2950439 L12.4056548,23.2595131 C12.3958229,23.2563662 12.3870493,23.2590235 12.3821421,23.2649074 L12.3780323,23.275831 L12.360941,23.7031097 L12.3658947,23.7234994 L12.3769048,23.7357139 L12.4804777,23.8096931 L12.4953491,23.8136134 L12.4953491,23.8136134 L12.5071152,23.8096931 L12.6106902,23.7357139 L12.6232938,23.7196733 L12.6232938,23.7196733 L12.6266527,23.7031097 L12.609561,23.275831 C12.6075724,23.2657013 12.6010112,23.2592993 12.5934901,23.257841 L12.5934901,23.257841 Z M12.8583906,23.1452862 L12.8445485,23.1473072 L12.6598443,23.2396597 L12.6498822,23.2499052 L12.6498822,23.2499052 L12.6471943,23.2611114 L12.6650943,23.6906389 L12.6699349,23.7034178 L12.6699349,23.7034178 L12.678386,23.7104931 L12.8793402,23.8032389 C12.8914285,23.8068999 12.9022333,23.8029875 12.9078286,23.7952264 L12.9118235,23.7811639 L12.8776777,23.1665331 C12.8752882,23.1545897 12.8674102,23.1470016 12.8583906,23.1452862 L12.8583906,23.1452862 Z M12.1430473,23.1473072 C12.1332178,23.1423925 12.1221763,23.1452606 12.1156365,23.1525954 L12.1099173,23.1665331 L12.0757714,23.7811639 C12.0751323,23.7926639 12.0828099,23.8018602 12.0926481,23.8045676 L12.108256,23.8032389 L12.3092106,23.7104931 L12.3186497,23.7024347 L12.3186497,23.7024347 L12.3225043,23.6906389 L12.340401,23.2611114 L12.337245,23.2485176 L12.337245,23.2485176 L12.3277531,23.2396597 L12.1430473,23.1473072 Z"
19
+ id="MingCute"
20
+ fillRule="nonzero"
21
+ ></path>
22
+ <path
23
+ d="M13.299,3.1477 L21.933,18.1022 C22.5103,19.1022 21.7887,20.3522 20.634,20.3522 L3.36601,20.3522 C2.21131,20.3522 1.48962,19.1022 2.06697,18.1022 L10.7009,3.14771 C11.2783,2.14771 12.7217,2.1477 13.299,3.1477 Z M12,15 C11.4477,15 11,15.4477 11,16 C11,16.5523 11.4477,17 12,17 C12.5523,17 13,16.5523 13,16 C13,15.4477 12.5523,15 12,15 Z M12,8 C11.48715,8 11.0644908,8.38604429 11.0067275,8.88337975 L11,9 L11,13 C11,13.5523 11.4477,14 12,14 C12.51285,14 12.9355092,13.613973 12.9932725,13.1166239 L13,13 L13,9 C13,8.44772 12.5523,8 12,8 Z"
24
+ id="shape"
25
+ fill="#09244B"
26
+ ></path>
27
+ </g>
28
+ </g>
29
+ </g>
30
+ </svg>
31
+ );
32
+ export default AlertIcon;
@@ -0,0 +1,36 @@
1
+ const DJLogo = props => (
2
+ <svg
3
+ width="30"
4
+ height="30"
5
+ style={{
6
+ marginRight: '10px',
7
+ marginBottom: '2px',
8
+ stroke: 'transparent',
9
+ strokeWidth: '0px',
10
+ }}
11
+ viewBox="0 0 83 83"
12
+ fill="none"
13
+ xmlns="http://www.w3.org/2000/svg"
14
+ xmlnsXlink="http://www.w3.org/1999/xlink"
15
+ >
16
+ <rect width="83" height="83" fill="url(#pattern0)" />
17
+ <rect width="83" height="83" stroke="black" />
18
+ <defs>
19
+ <pattern
20
+ id="pattern0"
21
+ patternContentUnits="objectBoundingBox"
22
+ width="1"
23
+ height="1"
24
+ >
25
+ <use xlinkHref="#image0_1_2" transform="scale(0.00364964)" />
26
+ </pattern>
27
+ <image
28
+ id="image0_1_2"
29
+ width="274"
30
+ height="274"
31
+ xlinkHref=""
32
+ />
33
+ </defs>
34
+ </svg>
35
+ );
36
+ export default DJLogo;
@@ -0,0 +1,21 @@
1
+ const DeleteIcon = props => (
2
+ <svg
3
+ className="feather feather-trash-2"
4
+ fill="none"
5
+ height="24"
6
+ stroke="currentColor"
7
+ strokeLinecap="round"
8
+ strokeLinejoin="round"
9
+ strokeWidth="2"
10
+ viewBox="0 0 24 24"
11
+ width="24"
12
+ xmlns="http://www.w3.org/2000/svg"
13
+ >
14
+ <polyline points="3 6 5 6 21 6" />
15
+ <path d="M19 6v14a2 2 0 0 1-2 2H7a2 2 0 0 1-2-2V6m3 0V4a2 2 0 0 1 2-2h4a2 2 0 0 1 2 2v2" />
16
+ <line x1="10" x2="10" y1="11" y2="17" />
17
+ <line x1="14" x2="14" y1="11" y2="17" />
18
+ </svg>
19
+ );
20
+
21
+ export default DeleteIcon;
@@ -0,0 +1,18 @@
1
+ const EditIcon = props => (
2
+ <svg
3
+ className="feather feather-edit"
4
+ fill="none"
5
+ height="24"
6
+ stroke="currentColor"
7
+ strokeLinecap="round"
8
+ strokeLinejoin="round"
9
+ strokeWidth="2"
10
+ viewBox="0 0 24 24"
11
+ width="24"
12
+ xmlns="http://www.w3.org/2000/svg"
13
+ >
14
+ <path d="M11 4H4a2 2 0 0 0-2 2v14a2 2 0 0 0 2 2h14a2 2 0 0 0 2-2v-7" />
15
+ <path d="M18.5 2.5a2.121 2.121 0 0 1 3 3L12 15l-4 1 1-4 9.5-9.5z" />
16
+ </svg>
17
+ );
18
+ export default EditIcon;
package/src/app/index.tsx CHANGED
@@ -10,6 +10,7 @@ import { BrowserRouter, Routes, Route } from 'react-router-dom';
10
10
  import { NamespacePage } from './pages/NamespacePage/Loadable';
11
11
  import { NodePage } from './pages/NodePage/Loadable';
12
12
  import { SQLBuilderPage } from './pages/SQLBuilderPage/Loadable';
13
+ import { AddEditNodePage } from './pages/AddEditNodePage/Loadable';
13
14
  import { NotFoundPage } from './pages/NotFoundPage/Loadable';
14
15
  import { LoginPage } from './pages/LoginPage';
15
16
  import { Root } from './pages/Root/Loadable';
@@ -43,6 +44,11 @@ export function App() {
43
44
  <>
44
45
  <Route path="nodes" key="nodes">
45
46
  <Route path=":name" element={<NodePage />} />
47
+ <Route
48
+ path=":name/edit"
49
+ key="edit"
50
+ element={<AddEditNodePage />}
51
+ />
46
52
  </Route>
47
53
 
48
54
  <Route path="/" element={<NamespacePage />} key="index" />
@@ -53,6 +59,18 @@ export function App() {
53
59
  key="namespaces"
54
60
  />
55
61
  </Route>
62
+ <Route path="create/:nodeType">
63
+ <Route
64
+ path=":initialNamespace"
65
+ key="create"
66
+ element={<AddEditNodePage />}
67
+ />
68
+ <Route
69
+ path=""
70
+ key="create"
71
+ element={<AddEditNodePage />}
72
+ />
73
+ </Route>
56
74
  <Route
57
75
  path="sql"
58
76
  key="sql"
@@ -0,0 +1,33 @@
1
+ /**
2
+ * A React Select component for use in Formik forms.
3
+ */
4
+ import { useField } from 'formik';
5
+ import Select from 'react-select';
6
+
7
+ export const FormikSelect = ({
8
+ selectOptions,
9
+ formikFieldName,
10
+ placeholder,
11
+ defaultValue,
12
+ style,
13
+ }) => {
14
+ // eslint-disable-next-line no-unused-vars
15
+ const [field, _, helpers] = useField(formikFieldName);
16
+ const { setValue } = helpers;
17
+
18
+ return (
19
+ <Select
20
+ className="SelectInput"
21
+ defaultValue={defaultValue}
22
+ options={selectOptions}
23
+ placeholder={placeholder}
24
+ onBlur={field.onBlur}
25
+ onChange={option => setValue(option.value)}
26
+ styles={style}
27
+ />
28
+ );
29
+ };
30
+
31
+ FormikSelect.defaultProps = {
32
+ placeholder: '',
33
+ };
@@ -0,0 +1,36 @@
1
+ /**
2
+ * A field for the full node name, which is generated based on the node's input
3
+ * namespace and display name.
4
+ */
5
+ import { useField, useFormikContext } from 'formik';
6
+ import { useEffect } from 'react';
7
+
8
+ export const FullNameField = props => {
9
+ const { values, setFieldValue } = useFormikContext();
10
+ const [field, meta] = useField(props);
11
+
12
+ useEffect(() => {
13
+ // Set the value of the node's full name based on its namespace and display name
14
+ if (values.namespace && values.display_name) {
15
+ setFieldValue(
16
+ props.name,
17
+ `${values.namespace}.${values.display_name
18
+ .toLowerCase()
19
+ .replace(/ /g, '_')}`,
20
+ );
21
+ }
22
+ }, [setFieldValue, props.name, values]);
23
+
24
+ return (
25
+ <>
26
+ <input
27
+ {...props}
28
+ {...field}
29
+ className="FullNameField"
30
+ disabled="disabled"
31
+ id="FullName"
32
+ />
33
+ {!!meta.touched && !!meta.error && <div>{meta.error}</div>}
34
+ </>
35
+ );
36
+ };
@@ -0,0 +1,16 @@
1
+ /**
2
+ * Asynchronously loads the component for the Node page
3
+ */
4
+
5
+ import * as React from 'react';
6
+ import { lazyLoad } from '../../../utils/loadable';
7
+
8
+ export const AddEditNodePage = () => {
9
+ return lazyLoad(
10
+ () => import('./index'),
11
+ module => module.AddEditNodePage,
12
+ {
13
+ fallback: <div></div>,
14
+ },
15
+ )();
16
+ };
@@ -0,0 +1,89 @@
1
+ /**
2
+ * SQL query input field, which consists of a CodeMirror SQL editor with autocompletion
3
+ * (for node names and columns) and syntax highlighting.
4
+ */
5
+ import React from 'react';
6
+ import { Field, useFormikContext } from 'formik';
7
+ import CodeMirror from '@uiw/react-codemirror';
8
+ import { langs } from '@uiw/codemirror-extensions-langs';
9
+
10
+ export const NodeQueryField = ({ djClient, value }) => {
11
+ const [schema, setSchema] = React.useState([]);
12
+ const formik = useFormikContext();
13
+ const sqlExt = langs.sql({ schema: schema });
14
+
15
+ const initialAutocomplete = async context => {
16
+ // Based on the parsed prefix, we load node names with that prefix
17
+ // into the autocomplete schema. At this stage we don't load the columns
18
+ // to save on unnecessary calls
19
+ const word = context.matchBefore(/[\.\w]*/);
20
+ const matches = await djClient.nodes(word.text);
21
+ matches.forEach(nodeName => {
22
+ if (schema[nodeName] === undefined) {
23
+ schema[nodeName] = [];
24
+ setSchema(schema);
25
+ }
26
+ });
27
+ };
28
+
29
+ const updateFormik = val => {
30
+ formik.setFieldValue('query', val);
31
+ };
32
+
33
+ const updateAutocomplete = async (value, _) => {
34
+ // If a particular node has been chosen, load the columns of that node into
35
+ // the autocomplete schema for column-level autocompletion
36
+ for (var nodeName in schema) {
37
+ if (
38
+ value.includes(nodeName) &&
39
+ (!schema.hasOwnProperty(nodeName) ||
40
+ (schema.hasOwnProperty(nodeName) && schema[nodeName].length === 0))
41
+ ) {
42
+ const nodeDetails = await djClient.node(nodeName);
43
+ schema[nodeName] = nodeDetails.columns.map(col => col.name);
44
+ setSchema(schema);
45
+ }
46
+ }
47
+ };
48
+
49
+ return (
50
+ <>
51
+ <Field
52
+ type="textarea"
53
+ style={{ display: 'none' }}
54
+ as="textarea"
55
+ name="query"
56
+ id="Query"
57
+ />
58
+ <div role="button" tabIndex={0} className="relative flex bg-[#282a36]">
59
+ <CodeMirror
60
+ id={'query'}
61
+ name={'query'}
62
+ extensions={[
63
+ sqlExt,
64
+ sqlExt.language.data.of({
65
+ autocomplete: initialAutocomplete,
66
+ }),
67
+ ]}
68
+ value={value}
69
+ options={{
70
+ theme: 'default',
71
+ lineNumbers: true,
72
+ }}
73
+ width="100%"
74
+ height="400px"
75
+ style={{
76
+ margin: '0 0 23px 0',
77
+ flex: 1,
78
+ fontSize: '150%',
79
+ textAlign: 'left',
80
+ }}
81
+ onChange={(value, viewUpdate) => {
82
+ updateFormik(value);
83
+ updateAutocomplete(value, viewUpdate);
84
+ }}
85
+ />
86
+ </div>
87
+ </>
88
+ );
89
+ };
@@ -0,0 +1,44 @@
1
+ import React from 'react';
2
+ import { render, screen } from '@testing-library/react';
3
+ import userEvent from '@testing-library/user-event';
4
+ import { Formik, Form } from 'formik';
5
+ import { FormikSelect } from '../FormikSelect';
6
+
7
+ describe('FormikSelect', () => {
8
+ const namespaces = [
9
+ { value: 'default', label: 'default' },
10
+ { value: 'basic', label: 'basic' },
11
+ { value: 'basic.one', label: 'basic.one' },
12
+ { value: 'basic.two', label: 'basic.two' },
13
+ ];
14
+
15
+ const setup = () => {
16
+ const utils = render(
17
+ <Formik initialValues={{ selectedOption: '' }} onSubmit={jest.fn()}>
18
+ <Form>
19
+ <FormikSelect
20
+ selectOptions={namespaces}
21
+ formikFieldName="namespace"
22
+ placeholder="Choose Namespace"
23
+ defaultValue={{
24
+ value: 'basic.one',
25
+ label: 'basic.one',
26
+ }}
27
+ />
28
+ </Form>
29
+ </Formik>,
30
+ );
31
+
32
+ const selectInput = screen.getByRole('combobox');
33
+ return {
34
+ ...utils,
35
+ selectInput,
36
+ };
37
+ };
38
+
39
+ it('renders the select component with provided options', () => {
40
+ setup();
41
+ userEvent.click(screen.getByRole('combobox')); // to open the dropdown
42
+ expect(screen.getByText('basic.one')).toBeInTheDocument();
43
+ });
44
+ });
@@ -0,0 +1,29 @@
1
+ import React from 'react';
2
+ import { render, screen } from '@testing-library/react';
3
+ import { Formik, Form } from 'formik';
4
+ import { FullNameField } from '../FullNameField';
5
+
6
+ describe('FullNameField', () => {
7
+ const setup = initialValues => {
8
+ return render(
9
+ <Formik initialValues={initialValues} onSubmit={jest.fn()}>
10
+ <Form>
11
+ <FullNameField name="full_name" />
12
+ </Form>
13
+ </Formik>,
14
+ );
15
+ };
16
+
17
+ it('generates the full name based on namespace and display name', () => {
18
+ setup({ namespace: 'cats', display_name: 'Jasper the Cat' });
19
+ const fullNameInput = screen.getByRole('textbox');
20
+ expect(fullNameInput.value).toBe('cats.jasper_the_cat');
21
+ });
22
+
23
+ it('does not set the full name if namespace or display name is missing', () => {
24
+ setup({ namespace: '', display_name: '' });
25
+
26
+ const fullNameInput = screen.getByRole('textbox');
27
+ expect(fullNameInput.value).toBe('');
28
+ });
29
+ });
@@ -0,0 +1,30 @@
1
+ import React from 'react';
2
+ import { render, screen } from '@testing-library/react';
3
+ import { Formik, Form } from 'formik';
4
+ import { NodeQueryField } from '../NodeQueryField';
5
+
6
+ describe('NodeQueryField', () => {
7
+ const mockDjClient = {
8
+ nodes: jest.fn(),
9
+ node: jest.fn(),
10
+ };
11
+ const renderWithFormik = (djClient = mockDjClient) => {
12
+ return render(
13
+ <Formik initialValues={{ query: '' }} onSubmit={jest.fn()}>
14
+ <Form>
15
+ <NodeQueryField djClient={djClient} />
16
+ </Form>
17
+ </Formik>,
18
+ );
19
+ };
20
+
21
+ afterEach(() => {
22
+ jest.clearAllMocks();
23
+ });
24
+
25
+ it('renders without crashing', () => {
26
+ renderWithFormik();
27
+ expect(screen.getByRole('textbox')).toBeInTheDocument();
28
+ expect(screen.getByRole('button')).toBeInTheDocument();
29
+ });
30
+ });
@@ -0,0 +1,84 @@
1
+ // Jest Snapshot v1, https://goo.gl/fbAQLP
2
+
3
+ exports[`AddEditNodePage Create node page renders with the selected nodeType and namespace 1`] = `HTMLCollection []`;
4
+
5
+ exports[`AddEditNodePage Verify create node page form failed submission 1`] = `
6
+ HTMLCollection [
7
+ <div
8
+ class="message alert"
9
+ >
10
+ <svg
11
+ height="2em"
12
+ version="1.1"
13
+ viewBox="0 0 24 24"
14
+ width="2em"
15
+ xmlns="http://www.w3.org/2000/svg"
16
+ >
17
+ <title>
18
+ alert_fill
19
+ </title>
20
+ <g
21
+ fill="none"
22
+ fill-rule="evenodd"
23
+ id="page-1"
24
+ stroke="none"
25
+ stroke-width="1"
26
+ >
27
+ <g
28
+ fill-rule="nonzero"
29
+ id="System"
30
+ transform="translate(-48.000000, -48.000000)"
31
+ >
32
+ <g
33
+ id="alert_fill"
34
+ transform="translate(48.000000, 48.000000)"
35
+ >
36
+ <path
37
+ d="M24,0 L24,24 L0,24 L0,0 L24,0 Z M12.5934901,23.257841 L12.5819402,23.2595131 L12.5108777,23.2950439 L12.4918791,23.2987469 L12.4918791,23.2987469 L12.4767152,23.2950439 L12.4056548,23.2595131 C12.3958229,23.2563662 12.3870493,23.2590235 12.3821421,23.2649074 L12.3780323,23.275831 L12.360941,23.7031097 L12.3658947,23.7234994 L12.3769048,23.7357139 L12.4804777,23.8096931 L12.4953491,23.8136134 L12.4953491,23.8136134 L12.5071152,23.8096931 L12.6106902,23.7357139 L12.6232938,23.7196733 L12.6232938,23.7196733 L12.6266527,23.7031097 L12.609561,23.275831 C12.6075724,23.2657013 12.6010112,23.2592993 12.5934901,23.257841 L12.5934901,23.257841 Z M12.8583906,23.1452862 L12.8445485,23.1473072 L12.6598443,23.2396597 L12.6498822,23.2499052 L12.6498822,23.2499052 L12.6471943,23.2611114 L12.6650943,23.6906389 L12.6699349,23.7034178 L12.6699349,23.7034178 L12.678386,23.7104931 L12.8793402,23.8032389 C12.8914285,23.8068999 12.9022333,23.8029875 12.9078286,23.7952264 L12.9118235,23.7811639 L12.8776777,23.1665331 C12.8752882,23.1545897 12.8674102,23.1470016 12.8583906,23.1452862 L12.8583906,23.1452862 Z M12.1430473,23.1473072 C12.1332178,23.1423925 12.1221763,23.1452606 12.1156365,23.1525954 L12.1099173,23.1665331 L12.0757714,23.7811639 C12.0751323,23.7926639 12.0828099,23.8018602 12.0926481,23.8045676 L12.108256,23.8032389 L12.3092106,23.7104931 L12.3186497,23.7024347 L12.3186497,23.7024347 L12.3225043,23.6906389 L12.340401,23.2611114 L12.337245,23.2485176 L12.337245,23.2485176 L12.3277531,23.2396597 L12.1430473,23.1473072 Z"
38
+ fill-rule="nonzero"
39
+ id="MingCute"
40
+ />
41
+ <path
42
+ d="M13.299,3.1477 L21.933,18.1022 C22.5103,19.1022 21.7887,20.3522 20.634,20.3522 L3.36601,20.3522 C2.21131,20.3522 1.48962,19.1022 2.06697,18.1022 L10.7009,3.14771 C11.2783,2.14771 12.7217,2.1477 13.299,3.1477 Z M12,15 C11.4477,15 11,15.4477 11,16 C11,16.5523 11.4477,17 12,17 C12.5523,17 13,16.5523 13,16 C13,15.4477 12.5523,15 12,15 Z M12,8 C11.48715,8 11.0644908,8.38604429 11.0067275,8.88337975 L11,9 L11,13 C11,13.5523 11.4477,14 12,14 C12.51285,14 12.9355092,13.613973 12.9932725,13.1166239 L13,13 L13,9 C13,8.44772 12.5523,8 12,8 Z"
43
+ fill="#09244B"
44
+ id="shape"
45
+ />
46
+ </g>
47
+ </g>
48
+ </g>
49
+ </svg>
50
+ Some columns in the primary key [] were not found
51
+ </div>,
52
+ ]
53
+ `;
54
+
55
+ exports[`AddEditNodePage Verify create node page user interaction and successful form submission 1`] = `
56
+ HTMLCollection [
57
+ <div
58
+ class="message success"
59
+ >
60
+ <svg
61
+ class="bi bi-check-circle-fill"
62
+ fill="currentColor"
63
+ height="25"
64
+ viewBox="0 0 16 16"
65
+ width="25"
66
+ xmlns="http://www.w3.org/2000/svg"
67
+ >
68
+ <path
69
+ d="M16 8A8 8 0 1 1 0 8a8 8 0 0 1 16 0zm-3.97-3.03a.75.75 0 0 0-1.08.022L7.477 9.417 5.384 7.323a.75.75 0 0 0-1.06 1.06L6.97 11.03a.75.75 0 0 0 1.079-.02l3.992-4.99a.75.75 0 0 0-.01-1.05z"
70
+ />
71
+ </svg>
72
+ Successfully created
73
+ dimension
74
+ node
75
+
76
+ <a
77
+ href="/nodes/default.special_forces_contractors"
78
+ >
79
+ default.special_forces_contractors
80
+ </a>
81
+ !
82
+ </div>,
83
+ ]
84
+ `;