@openneuro/app 4.29.8 → 4.30.0-alpha.0

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 (36) hide show
  1. package/.scss-lint.yml +11 -11
  2. package/maintenance.html +26 -20
  3. package/package.json +3 -3
  4. package/src/@types/custom.d.ts +6 -0
  5. package/src/index.html +14 -10
  6. package/src/scripts/datalad/routes/dataset-redirect.tsx +2 -0
  7. package/src/scripts/dataset/mutations/__tests__/update-permissions.spec.jsx +2 -1
  8. package/src/scripts/dataset/mutations/update-permissions.tsx +1 -9
  9. package/src/scripts/routes.tsx +17 -0
  10. package/src/scripts/users/__tests__/user-account-view.spec.tsx +152 -0
  11. package/src/scripts/users/__tests__/user-card.spec.tsx +110 -0
  12. package/src/scripts/users/__tests__/user-query.spec.tsx +65 -0
  13. package/src/scripts/users/__tests__/user-routes.spec.tsx +102 -0
  14. package/src/scripts/users/__tests__/user-tabs.spec.tsx +84 -0
  15. package/src/scripts/users/components/close-button.tsx +20 -0
  16. package/src/scripts/users/components/edit-button.tsx +20 -0
  17. package/src/scripts/users/components/edit-list.tsx +103 -0
  18. package/src/scripts/users/components/edit-string.tsx +90 -0
  19. package/src/scripts/users/components/editable-content.tsx +98 -0
  20. package/src/scripts/users/scss/editable-content.scss +15 -0
  21. package/src/scripts/users/scss/user-meta-blocks.scss +14 -0
  22. package/src/scripts/users/scss/useraccountview.module.scss +20 -0
  23. package/src/scripts/users/scss/usercard.module.scss +24 -0
  24. package/src/scripts/users/scss/usercontainer.module.scss +38 -0
  25. package/src/scripts/users/scss/usertabs.module.scss +63 -0
  26. package/src/scripts/users/user-account-view.tsx +142 -0
  27. package/src/scripts/users/user-card.tsx +86 -0
  28. package/src/scripts/users/user-container.tsx +49 -0
  29. package/src/scripts/users/user-datasets-view.tsx +53 -0
  30. package/src/scripts/users/user-notifications-view.tsx +11 -0
  31. package/src/scripts/users/user-query.tsx +76 -0
  32. package/src/scripts/users/user-routes.tsx +52 -0
  33. package/src/scripts/users/user-tabs.tsx +74 -0
  34. package/src/scripts/utils/__tests__/markdown.spec.tsx +1 -2
  35. package/src/scripts/utils/validationUtils.ts +8 -0
  36. package/src/scripts/validation/validation.tsx +11 -8
package/.scss-lint.yml CHANGED
@@ -1,15 +1,15 @@
1
- scss_files: 'src/sass/**/*.scss'
1
+ scss_files: "src/sass/**/*.scss"
2
2
 
3
3
  exclude:
4
- - 'src/sass/animate/**'
5
- - 'src/sass/bootstrap/**'
6
- - 'src/sass/font-awesome/**'
7
- - 'src/sass/fonts/**'
8
- - 'src/sass/_bootstrap-compass.scss'
9
- - 'src/sass/_bootstrap-mincer.scss'
10
- - 'src/sass/_bootstrap-sprockets.scss'
11
- - 'src/sass/_bootstrap.scss'
12
- - 'src/sass/react-select.scss'
4
+ - "src/sass/animate/**"
5
+ - "src/sass/bootstrap/**"
6
+ - "src/sass/font-awesome/**"
7
+ - "src/sass/fonts/**"
8
+ - "src/sass/_bootstrap-compass.scss"
9
+ - "src/sass/_bootstrap-mincer.scss"
10
+ - "src/sass/_bootstrap-sprockets.scss"
11
+ - "src/sass/_bootstrap.scss"
12
+ - "src/sass/react-select.scss"
13
13
 
14
14
  linters:
15
15
  BorderZero:
@@ -26,4 +26,4 @@ linters:
26
26
  enabled: false
27
27
 
28
28
  NestingDepth:
29
- enabled: false
29
+ enabled: false
package/maintenance.html CHANGED
@@ -1,23 +1,29 @@
1
1
  <!DOCTYPE html>
2
2
  <html>
3
+ <head>
4
+ <base href="/" />
5
+ <meta http-equiv="expires" content="0" />
6
+ <meta charset="utf-8" />
7
+ <title>OpenNeuro - Maintenance</title>
8
+ <link id="favicon" rel="icon" type="image/png" href="/favicon.ico" />
9
+ <link
10
+ href="https://fonts.googleapis.com/css?family=Open+Sans:300,400,700,300italic,400italic,700italic|Cabin:400,400italic"
11
+ rel="stylesheet"
12
+ type="text/css"
13
+ />
14
+ </head>
3
15
 
4
- <head>
5
- <base href="/">
6
- </base>
7
- <meta http-equiv="expires" content="0" />
8
- <meta charset="utf-8" />
9
- <title>OpenNeuro - Maintenance</title>
10
- <link id="favicon" rel="icon" type="image/png" href="/favicon.ico" />
11
- <link href='https://fonts.googleapis.com/css?family=Open+Sans:300,400,700,300italic,400italic,700italic|Cabin:400,400italic'
12
- rel='stylesheet' type='text/css' />
13
- </head>
14
-
15
- <body style="font-family: Cabin, sans-serif; color: #424242; display: flex; align-items: center; justify-content: center; flex-direction: column;">
16
- <img src="" />
17
- <div style="font-size: 75px;">Open<span style="color: #00505c;">Neuro</span></div>
18
- <h1 style="font-family: Open Sans, sans-serif; font-weight: 100;">
19
- OpenNeuro is undergoing maintenance and will return shortly.
20
- </h1>
21
- </body>
22
-
23
- </html>
16
+ <body
17
+ style="font-family: Cabin, sans-serif; color: #424242; display: flex; align-items: center; justify-content: center; flex-direction: column"
18
+ >
19
+ <img
20
+ src=""
21
+ />
22
+ <div style="font-size: 75px">
23
+ Open<span style="color: #00505c">Neuro</span>
24
+ </div>
25
+ <h1 style="font-family: Open Sans, sans-serif; font-weight: 100">
26
+ OpenNeuro is undergoing maintenance and will return shortly.
27
+ </h1>
28
+ </body>
29
+ </html>
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@openneuro/app",
3
- "version": "4.29.8",
3
+ "version": "4.30.0-alpha.0",
4
4
  "description": "React JS web frontend for the OpenNeuro platform.",
5
5
  "license": "MIT",
6
6
  "main": "public/client.js",
@@ -20,7 +20,7 @@
20
20
  "@emotion/react": "11.11.1",
21
21
  "@emotion/styled": "11.11.0",
22
22
  "@niivue/niivue": "0.45.1",
23
- "@openneuro/components": "^4.29.8",
23
+ "@openneuro/components": "^4.30.0-alpha.0",
24
24
  "@sentry/react": "^8.25.0",
25
25
  "@tanstack/react-table": "^8.9.3",
26
26
  "buffer": "^6.0.3",
@@ -74,5 +74,5 @@
74
74
  "publishConfig": {
75
75
  "access": "public"
76
76
  },
77
- "gitHead": "1cf2284b38246032cf3eca1046e697301d7c78e1"
77
+ "gitHead": "e310c9093b9acc5ff079664407cd4f2da9522651"
78
78
  }
@@ -11,6 +11,12 @@ declare module "*.svg" {
11
11
  export = value
12
12
  }
13
13
 
14
+ // Allow custom scss modules
15
+ declare module "*.module.scss" {
16
+ const classes: { [key: string]: string }
17
+ export default classes
18
+ }
19
+
14
20
  // Allow .scss imports
15
21
  declare module "*.scss" {
16
22
  const value: string
package/src/index.html CHANGED
@@ -1,23 +1,27 @@
1
1
  <!DOCTYPE html>
2
2
  <html>
3
-
4
- <head>
3
+ <head>
5
4
  <script src="/crn/config.js" type="module"></script>
6
5
  <script async src="https://www.googletagmanager.com/gtag/js"></script>
7
- <script src="https://kit.fontawesome.com/fa7ae96ba1.js" crossorigin="anonymous"></script>
6
+ <script
7
+ src="https://kit.fontawesome.com/fa7ae96ba1.js"
8
+ crossorigin="anonymous"
9
+ ></script>
8
10
  <meta http-equiv="expires" content="0" />
9
11
  <meta charset="utf-8" />
10
12
  <meta name="viewport" content="width=device-width, initial-scale=1" />
11
13
  <title>OpenNeuro</title>
12
14
  <link id="favicon" rel="icon" type="image/png" href="/favicon.ico" />
13
- <link rel="preconnect" href="https://fonts.googleapis.com">
14
- <link rel="preconnect" href="https://fonts.gstatic.com" crossorigin>
15
- <link href="https://fonts.googleapis.com/css2?family=Open+Sans:ital,wght@0,300;0,400;0,700;1,300;1,400;1,700&display=swap" rel="stylesheet">
16
- </head>
15
+ <link rel="preconnect" href="https://fonts.googleapis.com" />
16
+ <link rel="preconnect" href="https://fonts.gstatic.com" crossorigin />
17
+ <link
18
+ href="https://fonts.googleapis.com/css2?family=Open+Sans:ital,wght@0,300;0,400;0,700;1,300;1,400;1,700&display=swap"
19
+ rel="stylesheet"
20
+ />
21
+ </head>
17
22
 
18
- <body>
23
+ <body>
19
24
  <div id="main"></div>
20
25
  <script src="/client.jsx" type="module"></script>
21
- </body>
22
-
26
+ </body>
23
27
  </html>
@@ -49,6 +49,8 @@ const redirectLib = {
49
49
  ds002222: "ds002250",
50
50
  ds002245: "ds002345",
51
51
  ds001988: "ds001996",
52
+ ds004215: "ds005752",
53
+ ds004935: "ds005754",
52
54
  }
53
55
 
54
56
  const datasetIdPattern = /ds\d{6}/gim
@@ -2,12 +2,13 @@ import React from "react"
2
2
  import { fireEvent, render, screen, waitFor } from "@testing-library/react"
3
3
  import { MockedProvider } from "@apollo/client/testing"
4
4
  import {
5
- isValidOrcid,
6
5
  UPDATE_ORCID_PERMISSIONS,
7
6
  UPDATE_PERMISSIONS,
8
7
  UpdateDatasetPermissions,
9
8
  } from "../update-permissions"
10
9
 
10
+ import { isValidOrcid } from "../../../utils/validationUtils.ts"
11
+
11
12
  function permissionMocksFactory(
12
13
  updatePermissionsCalled,
13
14
  updateOrcidPermissionsCalled,
@@ -7,15 +7,7 @@ import ToastContent from "../../common/partials/toast-content"
7
7
  import { validate as isValidEmail } from "email-validator"
8
8
  import { Button } from "@openneuro/components/button"
9
9
 
10
- export function isValidOrcid(orcid: string) {
11
- if (orcid) {
12
- return /^[0-9]{4}-[0-9]{4}-[0-9]{4}-[0-9]{3}[0-9X]$/.test(orcid)
13
- ? true
14
- : false
15
- } else {
16
- return false
17
- }
18
- }
10
+ import { isValidOrcid } from "../../utils/validationUtils"
19
11
 
20
12
  export const UPDATE_PERMISSIONS = gql`
21
13
  mutation updatePermissions(
@@ -17,6 +17,10 @@ import FourOFourPage from "./errors/404page"
17
17
  import { ImportDataset } from "./pages/import-dataset"
18
18
  import { DatasetMetadata } from "./pages/metadata/dataset-metadata"
19
19
  import { TermsPage } from "./pages/terms"
20
+ import { UserQuery } from "./users/user-query"
21
+ import LoggedIn from "../scripts/authentication/logged-in"
22
+ import LoggedOut from "../scripts/authentication/logged-out"
23
+ import FourOThreePage from "./errors/403page"
20
24
 
21
25
  const AppRoutes: React.VoidFunctionComponent = () => (
22
26
  <Routes>
@@ -33,6 +37,19 @@ const AppRoutes: React.VoidFunctionComponent = () => (
33
37
  <Route path="/import" element={<ImportDataset />} />
34
38
  <Route path="/metadata" element={<DatasetMetadata />} />
35
39
  <Route path="/public" element={<Navigate to="/search" replace />} />
40
+ <Route
41
+ path="/user/:orcid/*"
42
+ element={
43
+ <>
44
+ <LoggedIn>
45
+ <UserQuery />
46
+ </LoggedIn>
47
+ <LoggedOut>
48
+ <FourOThreePage />
49
+ </LoggedOut>
50
+ </>
51
+ }
52
+ />
36
53
  <Route
37
54
  path="/saved"
38
55
  element={<Navigate to="/search?bookmarks" replace />}
@@ -0,0 +1,152 @@
1
+ import React from "react"
2
+ import { MockedProvider } from "@apollo/client/testing"
3
+ import {
4
+ fireEvent,
5
+ render,
6
+ screen,
7
+ waitFor,
8
+ within,
9
+ } from "@testing-library/react"
10
+ import { UserAccountView } from "../user-account-view"
11
+ import { GET_USER_BY_ORCID, UPDATE_USER } from "../user-query"
12
+
13
+ const baseUser = {
14
+ id: "1",
15
+ name: "John Doe",
16
+ email: "johndoe@example.com",
17
+ orcid: "0000-0001-2345-6789",
18
+ location: "San Francisco, CA",
19
+ institution: "University of California",
20
+ links: ["https://example.com", "https://example.org"],
21
+ github: "johndoe",
22
+ }
23
+
24
+ const mocks = [
25
+ {
26
+ request: {
27
+ query: GET_USER_BY_ORCID,
28
+ variables: { userId: baseUser.id },
29
+ },
30
+ result: {
31
+ data: {
32
+ user: baseUser,
33
+ },
34
+ },
35
+ },
36
+ {
37
+ request: {
38
+ query: UPDATE_USER,
39
+ variables: {
40
+ id: baseUser.id,
41
+ location: "Marin, CA",
42
+ links: ["https://newlink.com"],
43
+ institution: "New University",
44
+ },
45
+ },
46
+ result: {
47
+ data: {
48
+ updateUser: {
49
+ id: baseUser.id,
50
+ location: "Marin, CA",
51
+ links: ["https://newlink.com"],
52
+ institution: "New University",
53
+ },
54
+ },
55
+ },
56
+ },
57
+ ]
58
+
59
+ describe("<UserAccountView />", () => {
60
+ it("should render the user details correctly", () => {
61
+ render(
62
+ <MockedProvider mocks={mocks} addTypename={false}>
63
+ <UserAccountView user={baseUser} />
64
+ </MockedProvider>,
65
+ )
66
+ expect(screen.getByText("Name:")).toBeInTheDocument()
67
+ expect(screen.getByText("John Doe")).toBeInTheDocument()
68
+ expect(screen.getByText("Email:")).toBeInTheDocument()
69
+ expect(screen.getByText("johndoe@example.com")).toBeInTheDocument()
70
+ expect(screen.getByText("ORCID:")).toBeInTheDocument()
71
+ expect(screen.getByText("0000-0001-2345-6789")).toBeInTheDocument()
72
+ expect(screen.getByText("GitHub:")).toBeInTheDocument()
73
+ expect(screen.getByText("johndoe")).toBeInTheDocument()
74
+ })
75
+
76
+ it("should render location with EditableContent", async () => {
77
+ render(
78
+ <MockedProvider mocks={mocks} addTypename={false}>
79
+ <UserAccountView user={baseUser} />
80
+ </MockedProvider>,
81
+ )
82
+ const locationSection = within(screen.getByTestId("location-section"))
83
+ expect(screen.getByText("Location")).toBeInTheDocument()
84
+ const editButton = locationSection.getByText("Edit")
85
+ fireEvent.click(editButton)
86
+ const textbox = locationSection.getByRole("textbox")
87
+ fireEvent.change(textbox, { target: { value: "Marin, CA" } })
88
+ const saveButton = locationSection.getByText("Save")
89
+ fireEvent.click(saveButton)
90
+ await waitFor(() => {
91
+ expect(locationSection.getByText("Marin, CA")).toBeInTheDocument()
92
+ })
93
+ })
94
+
95
+ it("should render institution with EditableContent", async () => {
96
+ render(
97
+ <MockedProvider mocks={mocks} addTypename={false}>
98
+ <UserAccountView user={baseUser} />
99
+ </MockedProvider>,
100
+ )
101
+ const institutionSection = within(screen.getByTestId("institution-section"))
102
+ expect(screen.getByText("Institution")).toBeInTheDocument()
103
+ const editButton = institutionSection.getByText("Edit")
104
+ fireEvent.click(editButton)
105
+ const textbox = institutionSection.getByRole("textbox")
106
+ fireEvent.change(textbox, { target: { value: "New University" } })
107
+ const saveButton = institutionSection.getByText("Save")
108
+ fireEvent.click(saveButton)
109
+ await waitFor(() => {
110
+ expect(institutionSection.getByText("New University")).toBeInTheDocument()
111
+ })
112
+ })
113
+
114
+ it("should render links with EditableContent and validation", async () => {
115
+ render(
116
+ <MockedProvider mocks={mocks} addTypename={false}>
117
+ <UserAccountView user={baseUser} />
118
+ </MockedProvider>,
119
+ )
120
+ const linksSection = within(screen.getByTestId("links-section"))
121
+ expect(screen.getByText("Links")).toBeInTheDocument()
122
+ const editButton = linksSection.getByText("Edit")
123
+ fireEvent.click(editButton)
124
+ const textbox = linksSection.getByRole("textbox")
125
+ fireEvent.change(textbox, { target: { value: "https://newlink.com" } })
126
+ const saveButton = linksSection.getByText("Add")
127
+ fireEvent.click(saveButton)
128
+ await waitFor(() => {
129
+ expect(linksSection.getByText("https://newlink.com")).toBeInTheDocument()
130
+ })
131
+ })
132
+
133
+ it("should show an error message when invalid URL is entered in links section", async () => {
134
+ render(
135
+ <MockedProvider mocks={mocks} addTypename={false}>
136
+ <UserAccountView user={baseUser} />
137
+ </MockedProvider>,
138
+ )
139
+ const linksSection = within(screen.getByTestId("links-section"))
140
+ const editButton = linksSection.getByText("Edit")
141
+ fireEvent.click(editButton)
142
+ const textbox = linksSection.getByRole("textbox")
143
+ fireEvent.change(textbox, { target: { value: "invalid-url" } })
144
+ const saveButton = linksSection.getByText("Add")
145
+ fireEvent.click(saveButton)
146
+ await waitFor(() => {
147
+ expect(
148
+ linksSection.getByText("Invalid URL format. Please use a valid link."),
149
+ ).toBeInTheDocument()
150
+ })
151
+ })
152
+ })
@@ -0,0 +1,110 @@
1
+ import React from "react"
2
+ import { render, screen } from "@testing-library/react"
3
+ import type { User } from "../user-card"
4
+ import { UserCard } from "../user-card"
5
+
6
+ describe("UserCard Component", () => {
7
+ const baseUser: User = {
8
+ name: "John Doe",
9
+ email: "johndoe@example.com",
10
+ orcid: "0000-0001-2345-6789",
11
+ location: "San Francisco, CA",
12
+ institution: "University of California",
13
+ links: ["https://example.com", "https://example.org"],
14
+ github: "johndoe",
15
+ }
16
+
17
+ it("renders all user details when all data is provided", () => {
18
+ render(<UserCard user={baseUser} />)
19
+
20
+ const orcidLink = screen.getByRole("link", {
21
+ name: "ORCID profile of John Doe",
22
+ })
23
+ expect(orcidLink).toHaveAttribute(
24
+ "href",
25
+ "https://orcid.org/0000-0001-2345-6789",
26
+ )
27
+ expect(screen.getByText("University of California")).toBeInTheDocument()
28
+ expect(screen.getByText("San Francisco, CA")).toBeInTheDocument()
29
+
30
+ const emailLink = screen.getByRole("link", { name: "johndoe@example.com" })
31
+ expect(emailLink).toHaveAttribute("href", "mailto:johndoe@example.com")
32
+
33
+ const githubLink = screen.getByRole("link", {
34
+ name: "Github profile of John Doe",
35
+ })
36
+ expect(githubLink).toHaveAttribute("href", "https://github.com/johndoe")
37
+ expect(
38
+ screen.getByRole("link", { name: "https://example.com" }),
39
+ ).toHaveAttribute("href", "https://example.com")
40
+ expect(
41
+ screen.getByRole("link", { name: "https://example.org" }),
42
+ ).toHaveAttribute("href", "https://example.org")
43
+ })
44
+
45
+ it("renders without optional fields", () => {
46
+ const minimalUser: User = {
47
+ name: "Jane Doe",
48
+ email: "janedoe@example.com",
49
+ orcid: "0000-0002-3456-7890",
50
+ links: [],
51
+ }
52
+
53
+ render(<UserCard user={minimalUser} />)
54
+
55
+ const orcidLink = screen.getByRole("link", {
56
+ name: "ORCID profile of Jane Doe",
57
+ })
58
+ expect(orcidLink).toHaveAttribute(
59
+ "href",
60
+ "https://orcid.org/0000-0002-3456-7890",
61
+ )
62
+ const emailLink = screen.getByRole("link", { name: "janedoe@example.com" })
63
+ expect(emailLink).toHaveAttribute("href", "mailto:janedoe@example.com")
64
+ expect(screen.queryByText("University of California")).not
65
+ .toBeInTheDocument()
66
+ expect(screen.queryByText("San Francisco, CA")).not.toBeInTheDocument()
67
+ expect(screen.queryByRole("link", { name: "Github profile of Jane Doe" }))
68
+ .not.toBeInTheDocument()
69
+ })
70
+
71
+ it("renders correctly when links are empty", () => {
72
+ const userWithEmptyLinks: User = {
73
+ ...baseUser,
74
+ links: [],
75
+ }
76
+
77
+ render(<UserCard user={userWithEmptyLinks} />)
78
+
79
+ expect(screen.queryByRole("link", { name: "https://example.com" })).not
80
+ .toBeInTheDocument()
81
+ expect(screen.queryByRole("link", { name: "https://example.org" })).not
82
+ .toBeInTheDocument()
83
+ })
84
+
85
+ it("renders correctly when location and institution are missing", () => {
86
+ const userWithoutLocationAndInstitution: User = {
87
+ name: "Emily Doe",
88
+ email: "emilydoe@example.com",
89
+ orcid: "0000-0003-4567-8901",
90
+ links: ["https://example.com"],
91
+ }
92
+
93
+ render(<UserCard user={userWithoutLocationAndInstitution} />)
94
+
95
+ const orcidLink = screen.getByRole("link", {
96
+ name: "ORCID profile of Emily Doe",
97
+ })
98
+ expect(orcidLink).toHaveAttribute(
99
+ "href",
100
+ "https://orcid.org/0000-0003-4567-8901",
101
+ )
102
+ const emailLink = screen.getByRole("link", { name: "emilydoe@example.com" })
103
+ expect(emailLink).toHaveAttribute("href", "mailto:emilydoe@example.com")
104
+ const link = screen.getByRole("link", { name: "https://example.com" })
105
+ expect(link).toHaveAttribute("href", "https://example.com")
106
+ expect(screen.queryByText("San Francisco, CA")).not.toBeInTheDocument()
107
+ expect(screen.queryByText("University of California")).not
108
+ .toBeInTheDocument()
109
+ })
110
+ })