datajunction-ui 0.0.1-rc.2 → 0.0.1-rc.22

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.
Files changed (135) hide show
  1. package/.env +1 -0
  2. package/.prettierignore +3 -1
  3. package/dj-logo.svg +10 -0
  4. package/package.json +43 -13
  5. package/public/favicon.ico +0 -0
  6. package/src/__tests__/reportWebVitals.test.jsx +44 -0
  7. package/src/app/__tests__/__snapshots__/index.test.tsx.snap +5 -39
  8. package/src/app/components/DeleteNode.jsx +79 -0
  9. package/src/app/components/ListGroupItem.jsx +8 -1
  10. package/src/app/components/NamespaceHeader.jsx +4 -13
  11. package/src/app/components/QueryInfo.jsx +77 -0
  12. package/src/app/components/Tab.jsx +3 -2
  13. package/src/app/components/ToggleSwitch.jsx +20 -0
  14. package/src/app/components/__tests__/QueryInfo.test.jsx +55 -0
  15. package/src/app/components/__tests__/Tab.test.jsx +27 -0
  16. package/src/app/components/__tests__/ToggleSwitch.test.jsx +43 -0
  17. package/src/app/components/__tests__/__snapshots__/ListGroupItem.test.tsx.snap +3 -0
  18. package/src/app/components/__tests__/__snapshots__/NamespaceHeader.test.jsx.snap +2 -18
  19. package/src/app/components/djgraph/Collapse.jsx +46 -0
  20. package/src/app/components/djgraph/DJNode.jsx +60 -82
  21. package/src/app/components/djgraph/DJNodeColumns.jsx +71 -0
  22. package/src/app/components/djgraph/DJNodeDimensions.jsx +75 -0
  23. package/src/app/components/djgraph/LayoutFlow.jsx +104 -0
  24. package/src/app/components/djgraph/__tests__/Collapse.test.jsx +51 -0
  25. package/src/app/components/djgraph/__tests__/DJNodeColumns.test.jsx +83 -0
  26. package/src/app/components/djgraph/__tests__/DJNodeDimensions.test.jsx +118 -0
  27. package/src/app/components/djgraph/__tests__/__snapshots__/DJNode.test.tsx.snap +84 -40
  28. package/src/app/constants.js +2 -0
  29. package/src/app/icons/AlertIcon.jsx +32 -0
  30. package/src/app/icons/CollapsedIcon.jsx +15 -0
  31. package/src/app/icons/DJLogo.jsx +36 -0
  32. package/src/app/icons/DeleteIcon.jsx +21 -0
  33. package/src/app/icons/EditIcon.jsx +18 -0
  34. package/src/app/icons/ExpandedIcon.jsx +15 -0
  35. package/src/app/icons/HorizontalHierarchyIcon.jsx +15 -0
  36. package/src/app/icons/InvalidIcon.jsx +14 -0
  37. package/src/app/icons/PythonIcon.jsx +52 -0
  38. package/src/app/icons/TableIcon.jsx +14 -0
  39. package/src/app/icons/ValidIcon.jsx +14 -0
  40. package/src/app/index.tsx +79 -26
  41. package/src/app/pages/AddEditNodePage/FormikSelect.jsx +46 -0
  42. package/src/app/pages/AddEditNodePage/FullNameField.jsx +37 -0
  43. package/src/app/pages/AddEditNodePage/Loadable.jsx +16 -0
  44. package/src/app/pages/AddEditNodePage/NodeQueryField.jsx +89 -0
  45. package/src/app/pages/AddEditNodePage/__tests__/AddEditNodePageFormFailed.test.jsx +77 -0
  46. package/src/app/pages/AddEditNodePage/__tests__/AddEditNodePageFormSuccess.test.jsx +93 -0
  47. package/src/app/pages/AddEditNodePage/__tests__/FormikSelect.test.jsx +75 -0
  48. package/src/app/pages/AddEditNodePage/__tests__/FullNameField.test.jsx +31 -0
  49. package/src/app/pages/AddEditNodePage/__tests__/NodeQueryField.test.jsx +30 -0
  50. package/src/app/pages/AddEditNodePage/__tests__/__snapshots__/AddEditNodePageFormFailed.test.jsx.snap +53 -0
  51. package/src/app/pages/AddEditNodePage/__tests__/__snapshots__/AddEditNodePageFormSuccess.test.jsx.snap +53 -0
  52. package/src/app/pages/AddEditNodePage/__tests__/__snapshots__/index.test.jsx.snap +3 -0
  53. package/src/app/pages/AddEditNodePage/__tests__/index.test.jsx +178 -0
  54. package/src/app/pages/AddEditNodePage/index.jsx +357 -0
  55. package/src/app/pages/LoginPage/__tests__/index.test.jsx +70 -0
  56. package/src/app/pages/LoginPage/assets/sign-in-with-github.png +0 -0
  57. package/src/app/pages/LoginPage/assets/sign-in-with-google.png +0 -0
  58. package/src/app/pages/LoginPage/index.jsx +90 -0
  59. package/src/app/pages/NamespacePage/Explorer.jsx +57 -0
  60. package/src/app/pages/NamespacePage/Loadable.jsx +9 -7
  61. package/src/app/pages/NamespacePage/__tests__/index.test.jsx +95 -0
  62. package/src/app/pages/NamespacePage/index.jsx +131 -31
  63. package/src/app/pages/NodePage/ClientCodePopover.jsx +32 -0
  64. package/src/app/pages/NodePage/EditColumnPopover.jsx +102 -0
  65. package/src/app/pages/NodePage/LinkDimensionPopover.jsx +135 -0
  66. package/src/app/pages/NodePage/Loadable.jsx +9 -7
  67. package/src/app/pages/NodePage/NodeColumnTab.jsx +106 -27
  68. package/src/app/pages/NodePage/NodeGraphTab.jsx +94 -148
  69. package/src/app/pages/NodePage/NodeHistory.jsx +212 -0
  70. package/src/app/pages/NodePage/NodeInfoTab.jsx +166 -51
  71. package/src/app/pages/NodePage/NodeLineageTab.jsx +84 -0
  72. package/src/app/pages/NodePage/NodeMaterializationTab.jsx +174 -0
  73. package/src/app/pages/NodePage/NodeSQLTab.jsx +82 -0
  74. package/src/app/pages/NodePage/NodeStatus.jsx +14 -20
  75. package/src/app/pages/NodePage/NodesWithDimension.jsx +42 -0
  76. package/src/app/pages/NodePage/__tests__/ClientCodePopover.test.jsx +49 -0
  77. package/src/app/pages/NodePage/__tests__/EditColumnPopover.test.jsx +148 -0
  78. package/src/app/pages/NodePage/__tests__/LinkDimensionPopover.test.jsx +165 -0
  79. package/src/app/pages/NodePage/__tests__/NodeGraphTab.test.jsx +591 -0
  80. package/src/app/pages/NodePage/__tests__/NodeLineageTab.test.jsx +57 -0
  81. package/src/app/pages/NodePage/__tests__/NodePage.test.jsx +725 -0
  82. package/src/app/pages/NodePage/__tests__/NodeWithDimension.test.jsx +175 -0
  83. package/src/app/pages/NodePage/__tests__/__snapshots__/NodePage.test.jsx.snap +402 -0
  84. package/src/app/pages/NodePage/index.jsx +151 -41
  85. package/src/app/pages/NotFoundPage/__tests__/index.test.jsx +16 -0
  86. package/src/app/pages/RegisterTablePage/Loadable.jsx +16 -0
  87. package/src/app/pages/RegisterTablePage/index.jsx +163 -0
  88. package/src/app/pages/Root/__tests__/index.test.jsx +77 -0
  89. package/src/app/pages/Root/index.tsx +32 -4
  90. package/src/app/pages/SQLBuilderPage/Loadable.jsx +16 -0
  91. package/src/app/pages/SQLBuilderPage/__tests__/index.test.jsx +173 -0
  92. package/src/app/pages/SQLBuilderPage/index.jsx +390 -0
  93. package/src/app/providers/djclient.jsx +5 -0
  94. package/src/app/services/DJService.js +388 -22
  95. package/src/app/services/__tests__/DJService.test.jsx +609 -0
  96. package/src/mocks/mockNodes.jsx +1397 -0
  97. package/src/setupTests.ts +31 -1
  98. package/src/styles/dag.css +111 -5
  99. package/src/styles/index.css +467 -31
  100. package/src/styles/login.css +67 -0
  101. package/src/styles/node-creation.scss +197 -0
  102. package/src/styles/styles.scss +44 -0
  103. package/src/styles/styles.scss.d.ts +9 -0
  104. package/src/utils/form.jsx +23 -0
  105. package/tsconfig.json +1 -5
  106. package/webpack.config.js +29 -6
  107. package/.babelrc +0 -4
  108. package/.env.local +0 -4
  109. package/.env.production +0 -1
  110. package/.github/pull_request_template.md +0 -11
  111. package/.github/workflows/ci.yml +0 -33
  112. package/.vscode/extensions.json +0 -7
  113. package/.vscode/launch.json +0 -15
  114. package/.vscode/settings.json +0 -25
  115. package/Dockerfile +0 -7
  116. package/dist/5fa71a03d45dc2e3d61447f3013a303d.png +0 -0
  117. package/dist/index.html +0 -1
  118. package/dist/main.js +0 -23303
  119. package/dist/static/main.05a86d446163fd5f17d3.js +0 -2
  120. package/dist/static/main.05a86d446163fd5f17d3.js.LICENSE.txt +0 -98
  121. package/dist/static/main.9e53bed734dae98e5b10.js +0 -2
  122. package/dist/static/main.9e53bed734dae98e5b10.js.LICENSE.txt +0 -98
  123. package/dist/static/main.js +0 -2
  124. package/dist/static/main.js.LICENSE.txt +0 -98
  125. package/dist/static/vendor.05a86d446163fd5f17d3.js +0 -2
  126. package/dist/static/vendor.05a86d446163fd5f17d3.js.LICENSE.txt +0 -29
  127. package/dist/static/vendor.9e53bed734dae98e5b10.js +0 -2
  128. package/dist/static/vendor.9e53bed734dae98e5b10.js.LICENSE.txt +0 -29
  129. package/dist/static/vendor.js +0 -2
  130. package/dist/static/vendor.js.LICENSE.txt +0 -29
  131. package/dist/vendor.js +0 -281
  132. package/src/app/pages/ListNamespacesPage/Loadable.jsx +0 -14
  133. package/src/app/pages/ListNamespacesPage/index.jsx +0 -52
  134. package/src/app/pages/NamespacePage/__tests__/__snapshots__/index.test.tsx.snap +0 -45
  135. package/src/app/pages/NamespacePage/__tests__/index.test.tsx +0 -14
package/.env CHANGED
@@ -1 +1,2 @@
1
1
  REACT_APP_DJ_URL=http://localhost:8000
2
+ REACT_USE_SSE=true
package/.prettierignore CHANGED
@@ -1,4 +1,6 @@
1
1
  build/
2
2
  node_modules/
3
3
  package-lock.json
4
- yarn.lock
4
+ yarn.lock
5
+ dist/
6
+ coverage/
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.2",
3
+ "version": "0.0.1-rc.22",
4
4
  "description": "DataJunction Metrics Platform UI",
5
5
  "module": "src/index.tsx",
6
6
  "repository": {
@@ -23,49 +23,60 @@
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",
32
33
  "@types/react": "^18.0.20",
33
34
  "@types/react-dom": "^18.0.6",
34
35
  "@types/react-redux": "^7.1.24",
36
+ "@types/react-select": "5.0.1",
35
37
  "@types/react-test-renderer": "^18.0.0",
36
38
  "@types/rimraf": "^3.0.2",
37
39
  "@types/shelljs": "^0.8.11",
38
40
  "@types/testing-library__jest-dom": "^5.14.5",
39
41
  "@types/webpack": "^5.28.0",
40
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",
46
+ "babel-loader": "9.1.2",
41
47
  "chalk": "4.1.2",
48
+ "codemirror": "^6.0.0",
49
+ "cronstrue": "2.27.0",
42
50
  "cross-env": "7.0.3",
51
+ "css-loader": "6.8.1",
43
52
  "dagre": "^0.8.5",
44
- "eslint-config-prettier": "8.5.0",
45
- "eslint-plugin-prettier": "4.2.1",
46
- "eslint-plugin-react-hooks": "4.6.0",
47
- "fbemitter": "3.0.0",
48
- "flux": "4.0.4",
53
+ "datajunction": "0.0.1-rc.0",
54
+ "file-loader": "6.2.0",
49
55
  "fontfaceobserver": "2.3.0",
56
+ "formik": "2.4.3",
50
57
  "husky": "8.0.1",
51
58
  "i18next": "21.9.2",
52
59
  "i18next-browser-languagedetector": "6.1.5",
53
60
  "i18next-scanner": "4.0.0",
54
61
  "inquirer": "7.3.3",
55
62
  "inquirer-directory": "2.2.0",
63
+ "js-cookie": "3.0.5",
56
64
  "lint-staged": "13.0.3",
57
65
  "node-plop": "0.26.3",
58
66
  "plop": "2.7.6",
59
67
  "prettier": "2.7.1",
60
68
  "react": "18.2.0",
61
69
  "react-app-polyfill": "3.0.0",
70
+ "react-cookie": "4.1.1",
62
71
  "react-dom": "18.2.0",
63
72
  "react-helmet-async": "1.3.0",
64
73
  "react-i18next": "11.18.6",
65
74
  "react-is": "18.2.0",
75
+ "react-querybuilder": "6.5.1",
66
76
  "react-redux": "7.2.8",
67
77
  "react-router-dom": "6.3.0",
68
78
  "react-scripts": "5.0.1",
79
+ "react-select": "5.7.3",
69
80
  "react-syntax-highlighter": "^15.5.0",
70
81
  "react-test-renderer": "18.2.0",
71
82
  "reactflow": "^11.7.0",
@@ -73,9 +84,12 @@
73
84
  "redux-saga": "1.2.1",
74
85
  "rimraf": "3.0.2",
75
86
  "sanitize.css": "13.0.0",
87
+ "sass": "1.66.1",
88
+ "sass-loader": "13.3.2",
76
89
  "serve": "14.0.1",
77
90
  "shelljs": "0.8.5",
78
91
  "sql-formatter": "^12.2.0",
92
+ "style-loader": "3.3.3",
79
93
  "stylelint": "14.12.0",
80
94
  "stylelint-config-recommended": "9.0.0",
81
95
  "ts-loader": "9.4.2",
@@ -83,7 +97,8 @@
83
97
  "typescript": "4.6.4",
84
98
  "web-vitals": "2.1.4",
85
99
  "webpack": "5.81.0",
86
- "webpack-cli": "5.0.2"
100
+ "webpack-cli": "5.0.2",
101
+ "webpack-dev-server": "4.13.3"
87
102
  },
88
103
  "scripts": {
89
104
  "webpack-start": "webpack-dev-server --open",
@@ -132,6 +147,9 @@
132
147
  ]
133
148
  },
134
149
  "jest": {
150
+ "transformIgnorePatterns": [
151
+ "!node_modules/"
152
+ ],
135
153
  "collectCoverageFrom": [
136
154
  "src/**/*.{js,jsx,ts,tsx}",
137
155
  "!src/**/*/*.d.ts",
@@ -142,17 +160,29 @@
142
160
  ],
143
161
  "coverageThreshold": {
144
162
  "global": {
145
- "branches": 90,
146
- "functions": 90,
163
+ "statements": 90,
164
+ "branches": 75,
147
165
  "lines": 90,
148
- "statements": 90
166
+ "functions": 85
149
167
  }
150
168
  }
151
169
  },
170
+ "resolutions": {
171
+ "@codemirror/state": "6.2.0",
172
+ "@codemirror/view": "6.2.0",
173
+ "@lezer/common": "^1.0.0"
174
+ },
152
175
  "devDependencies": {
153
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",
179
+ "eslint-config-prettier": "8.8.0",
180
+ "eslint-plugin-prettier": "4.2.1",
181
+ "eslint-plugin-react-hooks": "4.6.0",
154
182
  "html-webpack-plugin": "5.5.1",
155
183
  "jest": "^29.5.0",
156
- "mini-css-extract-plugin": "2.7.5"
184
+ "jest-fetch-mock": "3.0.3",
185
+ "mini-css-extract-plugin": "2.7.6",
186
+ "resize-observer-polyfill": "1.5.1"
157
187
  }
158
188
  }
Binary file
@@ -0,0 +1,44 @@
1
+ import reportWebVitals from '../reportWebVitals';
2
+
3
+ // Mocking the web-vitals module
4
+ jest.mock('web-vitals', () => ({
5
+ getCLS: jest.fn(),
6
+ getFID: jest.fn(),
7
+ getFCP: jest.fn(),
8
+ getLCP: jest.fn(),
9
+ getTTFB: jest.fn(),
10
+ }));
11
+
12
+ describe('reportWebVitals', () => {
13
+ // Mock web-vitals functions
14
+ const mockGetCLS = jest.fn();
15
+ const mockGetFID = jest.fn();
16
+ const mockGetFCP = jest.fn();
17
+ const mockGetLCP = jest.fn();
18
+ const mockGetTTFB = jest.fn();
19
+
20
+ beforeAll(() => {
21
+ jest.doMock('web-vitals', () => ({
22
+ getCLS: mockGetCLS,
23
+ getFID: mockGetFID,
24
+ getFCP: mockGetFCP,
25
+ getLCP: mockGetLCP,
26
+ getTTFB: mockGetTTFB,
27
+ }));
28
+ });
29
+
30
+ beforeEach(() => {
31
+ jest.clearAllMocks();
32
+ });
33
+
34
+ it('does not call the web vitals functions if onPerfEntry is not a function', async () => {
35
+ await reportWebVitals(undefined);
36
+
37
+ const { getCLS, getFID, getFCP, getLCP, getTTFB } = require('web-vitals');
38
+ expect(getCLS).not.toHaveBeenCalled();
39
+ expect(getFID).not.toHaveBeenCalled();
40
+ expect(getFCP).not.toHaveBeenCalled();
41
+ expect(getLCP).not.toHaveBeenCalled();
42
+ expect(getTTFB).not.toHaveBeenCalled();
43
+ });
44
+ });
@@ -1,43 +1,9 @@
1
1
  // Jest Snapshot v1, https://goo.gl/fbAQLP
2
2
 
3
3
  exports[`<App /> should render and match the snapshot 1`] = `
4
- <BrowserRouter>
5
- <Helmet
6
- defaultTitle="DataJunction: A Metrics Platform"
7
- defer={true}
8
- encodeSpecialCharacters={true}
9
- prioritizeSeoTags={false}
10
- titleTemplate="DataJunction: %s"
11
- >
12
- <meta
13
- content="DataJunction serves as a semantic layer to help manage metrics"
14
- name="description"
15
- />
16
- </Helmet>
17
- <Routes>
18
- <Route
19
- element={<Unknown />}
20
- path="/"
21
- >
22
- <React.Fragment>
23
- <Route
24
- element={<Unknown />}
25
- path="nodes/:name"
26
- />
27
- <Route
28
- element={<ListNamespacesPage />}
29
- path="namespaces"
30
- />
31
- <Route
32
- element={<Unknown />}
33
- path="namespaces/:namespace"
34
- />
35
- </React.Fragment>
36
- </Route>
37
- <Route
38
- element={<Unknown />}
39
- path="*"
40
- />
41
- </Routes>
42
- </BrowserRouter>
4
+ <CookiesProvider>
5
+ <BrowserRouter>
6
+ <LoginPage />
7
+ </BrowserRouter>
8
+ </CookiesProvider>
43
9
  `;
@@ -0,0 +1,79 @@
1
+ import DJClientContext from '../providers/djclient';
2
+ import ValidIcon from '../icons/ValidIcon';
3
+ import AlertIcon from '../icons/AlertIcon';
4
+ import * as React from 'react';
5
+ import DeleteIcon from '../icons/DeleteIcon';
6
+ import { Form, Formik } from 'formik';
7
+ import { useContext } from 'react';
8
+
9
+ export default function DeleteNode({ nodeName }) {
10
+ console.log('nodeName', nodeName);
11
+
12
+ const djClient = useContext(DJClientContext).DataJunctionAPI;
13
+ const deleteNode = async (values, { setSubmitting, setStatus }) => {
14
+ const { status, json } = await djClient.deactivate(values.nodeName);
15
+ if (status === 200 || status === 201 || status === 204) {
16
+ setStatus({
17
+ success: <>Successfully deleted node {values.nodeName}</>,
18
+ });
19
+ } else {
20
+ setStatus({
21
+ failure: `${json.message}`,
22
+ });
23
+ }
24
+ setSubmitting(false);
25
+ };
26
+
27
+ const displayMessageAfterSubmit = status => {
28
+ return status?.success !== undefined ? (
29
+ <div className="message success">
30
+ <ValidIcon />
31
+ {status?.success}
32
+ </div>
33
+ ) : status?.failure !== undefined ? (
34
+ alertMessage(status?.failure)
35
+ ) : (
36
+ ''
37
+ );
38
+ };
39
+
40
+ const alertMessage = message => {
41
+ return (
42
+ <div className="message alert">
43
+ <AlertIcon />
44
+ {message}
45
+ </div>
46
+ );
47
+ };
48
+ const initialValues = {
49
+ nodeName: nodeName,
50
+ };
51
+
52
+ return (
53
+ <Formik initialValues={initialValues} onSubmit={deleteNode}>
54
+ {function Render({ isSubmitting, status, setFieldValue }) {
55
+ return (
56
+ <Form className="deleteNode">
57
+ {displayMessageAfterSubmit(status)}
58
+ {
59
+ <>
60
+ <button
61
+ type="submit"
62
+ disabled={isSubmitting}
63
+ style={{
64
+ marginLeft: 0,
65
+ all: 'unset',
66
+ color: '#005c72',
67
+ cursor: 'pointer',
68
+ }}
69
+ >
70
+ <DeleteIcon />
71
+ </button>
72
+ </>
73
+ }
74
+ </Form>
75
+ );
76
+ }}
77
+ </Formik>
78
+ );
79
+ }
@@ -8,7 +8,14 @@ export default class ListGroupItem extends Component {
8
8
  <div className="d-flex gap-2 w-100 justify-content-between py-3">
9
9
  <div>
10
10
  <h6 className="mb-0 w-100">{label}</h6>
11
- <p className="mb-0 opacity-75">{value}</p>
11
+ <p
12
+ className="mb-0 opacity-75"
13
+ role="dialog"
14
+ aria-hidden="false"
15
+ aria-label={label}
16
+ >
17
+ {value}
18
+ </p>
12
19
  </div>
13
20
  </div>
14
21
  </div>
@@ -1,4 +1,5 @@
1
1
  import { Component } from 'react';
2
+ import HorizontalHierarchyIcon from '../icons/HorizontalHierarchyIcon';
2
3
 
3
4
  export default class NamespaceHeader extends Component {
4
5
  render() {
@@ -6,7 +7,7 @@ export default class NamespaceHeader extends Component {
6
7
  const namespaceParts = namespace.split('.');
7
8
  const namespaceList = namespaceParts.map((piece, index) => {
8
9
  return (
9
- <li className="breadcrumb-item">
10
+ <li className="breadcrumb-item" key={index}>
10
11
  <a
11
12
  className="link-body-emphasis"
12
13
  href={'/namespaces/' + namespaceParts.slice(0, index + 1).join('.')}
@@ -19,18 +20,8 @@ export default class NamespaceHeader extends Component {
19
20
  return (
20
21
  <ol className="breadcrumb breadcrumb-chevron p-3 bg-body-tertiary rounded-3">
21
22
  <li className="breadcrumb-item">
22
- <a href="/namespaces/">
23
- <svg
24
- xmlns="http://www.w3.org/2000/svg"
25
- width="16"
26
- height="16"
27
- fill="currentColor"
28
- className="bi bi-house-door-fill"
29
- viewBox="0 0 16 16"
30
- style={{ paddingBottom: '0.2rem' }}
31
- >
32
- <path d="M6.5 14.5v-3.505c0-.245.25-.495.5-.495h2c.25 0 .5.25.5.5v3.5a.5.5 0 0 0 .5.5h4a.5.5 0 0 0 .5-.5v-7a.5.5 0 0 0-.146-.354L13 5.793V2.5a.5.5 0 0 0-.5-.5h-1a.5.5 0 0 0-.5.5v1.293L8.354 1.146a.5.5 0 0 0-.708 0l-6 6A.5.5 0 0 0 1.5 7.5v7a.5.5 0 0 0 .5.5h4a.5.5 0 0 0 .5-.5Z" />
33
- </svg>
23
+ <a href="/">
24
+ <HorizontalHierarchyIcon />
34
25
  </a>
35
26
  </li>
36
27
  {namespaceList}
@@ -0,0 +1,77 @@
1
+ export default function QueryInfo({
2
+ id,
3
+ state,
4
+ engine_name,
5
+ engine_version,
6
+ errors,
7
+ links,
8
+ output_table,
9
+ scheduled,
10
+ started,
11
+ numRows,
12
+ }) {
13
+ return (
14
+ <div className="table-responsive">
15
+ <table className="card-inner-table table">
16
+ <thead className="fs-7 fw-bold text-gray-400 border-bottom-0">
17
+ <tr>
18
+ <th>Query ID</th>
19
+ <th>Engine</th>
20
+ <th>State</th>
21
+ <th>Scheduled</th>
22
+ <th>Started</th>
23
+ <th>Errors</th>
24
+ <th>Links</th>
25
+ <th>Output Table</th>
26
+ <th>Number of Rows</th>
27
+ </tr>
28
+ </thead>
29
+ <tbody>
30
+ <tr>
31
+ <td>
32
+ <span className="rounded-pill badge bg-secondary-soft">{id}</span>
33
+ </td>
34
+ <td>
35
+ <span className="rounded-pill badge bg-secondary-soft">
36
+ {engine_name}
37
+ {' - '}
38
+ {engine_version}
39
+ </span>
40
+ </td>
41
+ <td>{state}</td>
42
+ <td>{scheduled}</td>
43
+ <td>{started}</td>
44
+ <td>
45
+ {errors?.length ? (
46
+ errors.map((e, idx) => (
47
+ <p key={`error-${idx}`}>
48
+ <span className="rounded-pill badge bg-secondary-error">
49
+ {e}
50
+ </span>
51
+ </p>
52
+ ))
53
+ ) : (
54
+ <></>
55
+ )}
56
+ </td>
57
+ <td>
58
+ {links?.length ? (
59
+ links.map((link, idx) => (
60
+ <p key={idx}>
61
+ <a href={link} target="_blank" rel="noreferrer">
62
+ {link}
63
+ </a>
64
+ </p>
65
+ ))
66
+ ) : (
67
+ <></>
68
+ )}
69
+ </td>
70
+ <td>{output_table}</td>
71
+ <td>{numRows}</td>
72
+ </tr>
73
+ </tbody>
74
+ </table>
75
+ </div>
76
+ );
77
+ }
@@ -9,13 +9,14 @@ export default class Tab extends Component {
9
9
  <div className="nav-item">
10
10
  <button
11
11
  id={id}
12
- role="button"
13
12
  className="nav-link"
14
13
  tabIndex="0"
15
14
  onClick={onClick}
15
+ role="button"
16
+ aria-label={this.props.name}
17
+ aria-hidden="false"
16
18
  >
17
19
  {this.props.name}
18
- {/*<span className="rounded-pill badge bg-secondary-soft">823</span>*/}
19
20
  </button>
20
21
  </div>
21
22
  </div>
@@ -0,0 +1,20 @@
1
+ import React from 'react';
2
+
3
+ const ToggleSwitch = ({ checked, onChange, toggleName }) => (
4
+ <>
5
+ <input
6
+ id="show-compiled-sql-toggle"
7
+ role="checkbox"
8
+ aria-label="ToggleSwitch"
9
+ aria-hidden="false"
10
+ type="checkbox"
11
+ className="checkbox"
12
+ checked={checked}
13
+ onChange={e => onChange(e.target.checked)}
14
+ />
15
+ <label htmlFor="show-compiled-sql-toggle" className="switch"></label>{' '}
16
+ {toggleName}
17
+ </>
18
+ );
19
+
20
+ export default ToggleSwitch;
@@ -0,0 +1,55 @@
1
+ import * as React from 'react';
2
+ import { render, screen } from '@testing-library/react';
3
+ import QueryInfo from '../QueryInfo';
4
+
5
+ describe('<QueryInfo />', () => {
6
+ const defaultProps = {
7
+ id: '123',
8
+ state: 'Running',
9
+ engine_name: 'Spark SQL',
10
+ engine_version: '1.0',
11
+ errors: ['Error 1', 'Error 2'],
12
+ links: ['http://example.com', 'http://example2.com'],
13
+ output_table: 'table1',
14
+ scheduled: '2023-09-06',
15
+ started: '2023-09-07',
16
+ numRows: 1000,
17
+ };
18
+
19
+ it('renders without crashing', () => {
20
+ render(<QueryInfo {...defaultProps} />);
21
+ });
22
+
23
+ it('displays correct query information', () => {
24
+ render(<QueryInfo {...defaultProps} />);
25
+
26
+ expect(screen.getByText(defaultProps.id)).toBeInTheDocument();
27
+ expect(
28
+ screen.getByText(
29
+ `${defaultProps.engine_name} - ${defaultProps.engine_version}`,
30
+ ),
31
+ ).toBeInTheDocument();
32
+ expect(screen.getByText(defaultProps.state)).toBeInTheDocument();
33
+ expect(screen.getByText(defaultProps.scheduled)).toBeInTheDocument();
34
+ expect(screen.getByText(defaultProps.started)).toBeInTheDocument();
35
+ expect(screen.getByText(defaultProps.output_table)).toBeInTheDocument();
36
+ expect(screen.getByText(String(defaultProps.numRows))).toBeInTheDocument();
37
+ defaultProps.errors.forEach(error => {
38
+ expect(screen.getByText(error)).toBeInTheDocument();
39
+ });
40
+ defaultProps.links.forEach(link => {
41
+ expect(screen.getByText(link)).toHaveAttribute('href', link);
42
+ });
43
+ });
44
+
45
+ it('does not render errors and links when they are not provided', () => {
46
+ render(<QueryInfo {...defaultProps} errors={[]} links={[]} />);
47
+
48
+ defaultProps.errors.forEach(error => {
49
+ expect(screen.queryByText(error)).not.toBeInTheDocument();
50
+ });
51
+ defaultProps.links.forEach(link => {
52
+ expect(screen.queryByText(link)).not.toBeInTheDocument();
53
+ });
54
+ });
55
+ });
@@ -0,0 +1,27 @@
1
+ import * as React from 'react';
2
+ import { render, fireEvent } from '@testing-library/react';
3
+
4
+ import Tab from '../Tab';
5
+
6
+ describe('<Tab />', () => {
7
+ it('renders without crashing', () => {
8
+ render(<Tab />);
9
+ });
10
+
11
+ it('has the active class when selectedTab matches id', () => {
12
+ const { container } = render(<Tab id="1" selectedTab="1" />);
13
+ expect(container.querySelector('.col')).toHaveClass('active');
14
+ });
15
+
16
+ it('does not have the active class when selectedTab does not match id', () => {
17
+ const { container } = render(<Tab id="1" selectedTab="2" />);
18
+ expect(container.querySelector('.col')).not.toHaveClass('active');
19
+ });
20
+
21
+ it('calls onClick when the button is clicked', () => {
22
+ const onClickMock = jest.fn();
23
+ const { getByRole } = render(<Tab id="1" onClick={onClickMock} />);
24
+ fireEvent.click(getByRole('button'));
25
+ expect(onClickMock).toHaveBeenCalledTimes(1);
26
+ });
27
+ });
@@ -0,0 +1,43 @@
1
+ import React from 'react';
2
+ import { render, fireEvent, screen } from '@testing-library/react';
3
+ import ToggleSwitch from '../ToggleSwitch';
4
+
5
+ describe('<ToggleSwitch />', () => {
6
+ const defaultProps = {
7
+ checked: false,
8
+ onChange: jest.fn(),
9
+ toggleName: 'Toggle Switch',
10
+ };
11
+
12
+ it('renders without crashing', () => {
13
+ render(<ToggleSwitch {...defaultProps} />);
14
+ });
15
+
16
+ it('displays the correct toggle name', () => {
17
+ render(<ToggleSwitch {...defaultProps} />);
18
+ expect(screen.getByText(defaultProps.toggleName)).toBeInTheDocument();
19
+ });
20
+
21
+ it('reflects the checked state correctly', () => {
22
+ render(<ToggleSwitch {...defaultProps} checked={true} />);
23
+ const checkbox = screen.getByRole('checkbox');
24
+ expect(checkbox).toBeChecked();
25
+ });
26
+
27
+ it('calls onChange with the correct value when toggled', () => {
28
+ render(<ToggleSwitch {...defaultProps} />);
29
+ const checkbox = screen.getByRole('checkbox');
30
+
31
+ fireEvent.click(checkbox);
32
+ expect(defaultProps.onChange).toHaveBeenCalledWith(true);
33
+
34
+ fireEvent.click(checkbox);
35
+ expect(checkbox).not.toBeChecked();
36
+ });
37
+
38
+ it('is unchecked by default if no checked prop is provided', () => {
39
+ render(<ToggleSwitch onChange={jest.fn()} toggleName="Test Toggle" />);
40
+ const checkbox = screen.getByRole('checkbox');
41
+ expect(checkbox).not.toBeChecked();
42
+ });
43
+ });