Merge branch 'fe/feat/141-workspace-browse-api' into 'fe/develop'

Feat: 워크스페이스 api 연결 - S11P21S002-141

See merge request s11-s-project/S11P21S002!59
This commit is contained in:
김태수 2024-09-11 15:10:50 +09:00
commit 33a9c74026
23 changed files with 2286 additions and 216 deletions

View File

@ -54,6 +54,7 @@
"eslint-plugin-react-hooks": "^4.6.2",
"eslint-plugin-react-refresh": "^0.4.7",
"eslint-plugin-storybook": "^0.8.0",
"msw": "^2.4.4",
"postcss": "^8.4.41",
"prettier": "^3.3.3",
"prettier-plugin-tailwindcss": "^0.6.6",
@ -2163,6 +2164,43 @@
"integrity": "sha512-4iri8i1AqYHJE2DstZYkyEprg6Pq6sKx3xn5FpySk9sNhH7qN2LLlHJCfDTZRILNwQNPD7mATWM0TBui7uC1pA==",
"dev": true
},
"node_modules/@bundled-es-modules/cookie": {
"version": "2.0.0",
"resolved": "https://registry.npmjs.org/@bundled-es-modules/cookie/-/cookie-2.0.0.tgz",
"integrity": "sha512-Or6YHg/kamKHpxULAdSqhGqnWFneIXu1NKvvfBBzKGwpVsYuFIQ5aBPHDnnoR3ghW1nvSkALd+EF9iMtY7Vjxw==",
"dev": true,
"dependencies": {
"cookie": "^0.5.0"
}
},
"node_modules/@bundled-es-modules/cookie/node_modules/cookie": {
"version": "0.5.0",
"resolved": "https://registry.npmjs.org/cookie/-/cookie-0.5.0.tgz",
"integrity": "sha512-YZ3GUyn/o8gfKJlnlX7g7xq4gyO6OSuhGPKaaGssGB2qgDUS0gPgtTvoyZLTt9Ab6dC4hfc9dV5arkvc/OCmrw==",
"dev": true,
"engines": {
"node": ">= 0.6"
}
},
"node_modules/@bundled-es-modules/statuses": {
"version": "1.0.1",
"resolved": "https://registry.npmjs.org/@bundled-es-modules/statuses/-/statuses-1.0.1.tgz",
"integrity": "sha512-yn7BklA5acgcBr+7w064fGV+SGIFySjCKpqjcWgBAIfrAkY+4GQTJJHQMeT3V/sgz23VTEVV8TtOmkvJAhFVfg==",
"dev": true,
"dependencies": {
"statuses": "^2.0.1"
}
},
"node_modules/@bundled-es-modules/tough-cookie": {
"version": "0.1.6",
"resolved": "https://registry.npmjs.org/@bundled-es-modules/tough-cookie/-/tough-cookie-0.1.6.tgz",
"integrity": "sha512-dvMHbL464C0zI+Yqxbz6kZ5TOEp7GLW+pry/RWndAR8MJQAXZ2rPmIs8tziTZjeIyhSNZgZbCePtfSbdWqStJw==",
"dev": true,
"dependencies": {
"@types/tough-cookie": "^4.0.5",
"tough-cookie": "^4.1.4"
}
},
"node_modules/@chromatic-com/storybook": {
"version": "1.7.0",
"resolved": "https://registry.npmjs.org/@chromatic-com/storybook/-/storybook-1.7.0.tgz",
@ -2791,6 +2829,98 @@
"dev": true,
"license": "BSD-3-Clause"
},
"node_modules/@inquirer/confirm": {
"version": "3.2.0",
"resolved": "https://registry.npmjs.org/@inquirer/confirm/-/confirm-3.2.0.tgz",
"integrity": "sha512-oOIwPs0Dvq5220Z8lGL/6LHRTEr9TgLHmiI99Rj1PJ1p1czTys+olrgBqZk4E2qC0YTzeHprxSQmoHioVdJ7Lw==",
"dev": true,
"dependencies": {
"@inquirer/core": "^9.1.0",
"@inquirer/type": "^1.5.3"
},
"engines": {
"node": ">=18"
}
},
"node_modules/@inquirer/core": {
"version": "9.1.0",
"resolved": "https://registry.npmjs.org/@inquirer/core/-/core-9.1.0.tgz",
"integrity": "sha512-RZVfH//2ytTjmaBIzeKT1zefcQZzuruwkpTwwbe/i2jTl4o9M+iML5ChULzz6iw1Ok8iUBBsRCjY2IEbD8Ft4w==",
"dev": true,
"dependencies": {
"@inquirer/figures": "^1.0.5",
"@inquirer/type": "^1.5.3",
"@types/mute-stream": "^0.0.4",
"@types/node": "^22.5.2",
"@types/wrap-ansi": "^3.0.0",
"ansi-escapes": "^4.3.2",
"cli-spinners": "^2.9.2",
"cli-width": "^4.1.0",
"mute-stream": "^1.0.0",
"signal-exit": "^4.1.0",
"strip-ansi": "^6.0.1",
"wrap-ansi": "^6.2.0",
"yoctocolors-cjs": "^2.1.2"
},
"engines": {
"node": ">=18"
}
},
"node_modules/@inquirer/core/node_modules/emoji-regex": {
"version": "8.0.0",
"resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz",
"integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==",
"dev": true
},
"node_modules/@inquirer/core/node_modules/string-width": {
"version": "4.2.3",
"resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz",
"integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==",
"dev": true,
"dependencies": {
"emoji-regex": "^8.0.0",
"is-fullwidth-code-point": "^3.0.0",
"strip-ansi": "^6.0.1"
},
"engines": {
"node": ">=8"
}
},
"node_modules/@inquirer/core/node_modules/wrap-ansi": {
"version": "6.2.0",
"resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-6.2.0.tgz",
"integrity": "sha512-r6lPcBGxZXlIcymEu7InxDMhdW0KDxpLgoFLcguasxCaJ/SOIZwINatK9KY/tf+ZrlywOKU0UDj3ATXUBfxJXA==",
"dev": true,
"dependencies": {
"ansi-styles": "^4.0.0",
"string-width": "^4.1.0",
"strip-ansi": "^6.0.0"
},
"engines": {
"node": ">=8"
}
},
"node_modules/@inquirer/figures": {
"version": "1.0.5",
"resolved": "https://registry.npmjs.org/@inquirer/figures/-/figures-1.0.5.tgz",
"integrity": "sha512-79hP/VWdZ2UVc9bFGJnoQ/lQMpL74mGgzSYX1xUqCVk7/v73vJCMw1VuyWN1jGkZ9B3z7THAbySqGbCNefcjfA==",
"dev": true,
"engines": {
"node": ">=18"
}
},
"node_modules/@inquirer/type": {
"version": "1.5.3",
"resolved": "https://registry.npmjs.org/@inquirer/type/-/type-1.5.3.tgz",
"integrity": "sha512-xUQ14WQGR/HK5ei+2CvgcwoH9fQ4PgPGmVFSN0pc1+fVyDL3MREhyAY7nxEErSu6CkllBM3D7e3e+kOvtu+eIg==",
"dev": true,
"dependencies": {
"mute-stream": "^1.0.0"
},
"engines": {
"node": ">=18"
}
},
"node_modules/@isaacs/cliui": {
"version": "8.0.2",
"resolved": "https://registry.npmjs.org/@isaacs/cliui/-/cliui-8.0.2.tgz",
@ -2945,6 +3075,23 @@
"react": ">=16"
}
},
"node_modules/@mswjs/interceptors": {
"version": "0.35.0",
"resolved": "https://registry.npmjs.org/@mswjs/interceptors/-/interceptors-0.35.0.tgz",
"integrity": "sha512-f5cHyIvm4m4g1I5x9EH1etGx0puaU0OaX2szqGRVBVgUC6aMASlOI5hbpe7tJ9l4/VWjCUu5OMraCazLZGI24A==",
"dev": true,
"dependencies": {
"@open-draft/deferred-promise": "^2.2.0",
"@open-draft/logger": "^0.3.0",
"@open-draft/until": "^2.0.0",
"is-node-process": "^1.2.0",
"outvariant": "^1.4.3",
"strict-event-emitter": "^0.5.1"
},
"engines": {
"node": ">=18"
}
},
"node_modules/@nodelib/fs.scandir": {
"version": "2.1.5",
"resolved": "https://registry.npmjs.org/@nodelib/fs.scandir/-/fs.scandir-2.1.5.tgz",
@ -2980,6 +3127,28 @@
"node": ">= 8"
}
},
"node_modules/@open-draft/deferred-promise": {
"version": "2.2.0",
"resolved": "https://registry.npmjs.org/@open-draft/deferred-promise/-/deferred-promise-2.2.0.tgz",
"integrity": "sha512-CecwLWx3rhxVQF6V4bAgPS5t+So2sTbPgAzafKkVizyi7tlwpcFpdFqq+wqF2OwNBmqFuu6tOyouTuxgpMfzmA==",
"dev": true
},
"node_modules/@open-draft/logger": {
"version": "0.3.0",
"resolved": "https://registry.npmjs.org/@open-draft/logger/-/logger-0.3.0.tgz",
"integrity": "sha512-X2g45fzhxH238HKO4xbSr7+wBS8Fvw6ixhTDuvLd5mqh6bJJCFAPwU9mPDxbcrRtfxv4u5IHCEH77BmxvXmmxQ==",
"dev": true,
"dependencies": {
"is-node-process": "^1.2.0",
"outvariant": "^1.4.0"
}
},
"node_modules/@open-draft/until": {
"version": "2.1.0",
"resolved": "https://registry.npmjs.org/@open-draft/until/-/until-2.1.0.tgz",
"integrity": "sha512-U69T3ItWHvLwGg5eJ0n3I62nWuE6ilHlmz7zM0npLBRvPRd7e6NYmg54vvRtP5mZG7kZqZCFVdsTWo7BPtBujg==",
"dev": true
},
"node_modules/@pkgjs/parseargs": {
"version": "0.11.0",
"resolved": "https://registry.npmjs.org/@pkgjs/parseargs/-/parseargs-0.11.0.tgz",
@ -5460,6 +5629,12 @@
"@types/node": "*"
}
},
"node_modules/@types/cookie": {
"version": "0.6.0",
"resolved": "https://registry.npmjs.org/@types/cookie/-/cookie-0.6.0.tgz",
"integrity": "sha512-4Kh9a6B2bQciAhf7FSuMRRkUWecJgJu9nPnx3yzpsfXX/c50REIqpHY4C82bXP90qrLtXtkDxTZosYO3UpOwlA==",
"dev": true
},
"node_modules/@types/cross-spawn": {
"version": "6.0.6",
"resolved": "https://registry.npmjs.org/@types/cross-spawn/-/cross-spawn-6.0.6.tgz",
@ -5579,12 +5754,20 @@
"integrity": "sha512-K0VQKziLUWkVKiRVrx4a40iPaxTUefQmjtkQofBkYRcoaaL/8rhwDWww9qWbrgicNOgnpIsMxyNIUM4+n6dUIA==",
"dev": true
},
"node_modules/@types/node": {
"version": "22.5.0",
"resolved": "https://registry.npmjs.org/@types/node/-/node-22.5.0.tgz",
"integrity": "sha512-DkFrJOe+rfdHTqqMg0bSNlGlQ85hSoh2TPzZyhHsXnMtligRWpxUySiyw8FY14ITt24HVCiQPWxS3KO/QlGmWg==",
"node_modules/@types/mute-stream": {
"version": "0.0.4",
"resolved": "https://registry.npmjs.org/@types/mute-stream/-/mute-stream-0.0.4.tgz",
"integrity": "sha512-CPM9nzrCPPJHQNA9keH9CVkVI+WR5kMa+7XEs5jcGQ0VoAGnLv242w8lIVgwAEfmE4oufJRaTc9PNLQl0ioAow==",
"dev": true,
"dependencies": {
"@types/node": "*"
}
},
"node_modules/@types/node": {
"version": "22.5.4",
"resolved": "https://registry.npmjs.org/@types/node/-/node-22.5.4.tgz",
"integrity": "sha512-FDuKUJQm/ju9fT/SeX/6+gBzoPzlVCzfzmGkwKvRHQVxi4BntVbyIwf6a4Xn62mrvndLiml6z/UBXIdEVjQLXg==",
"dev": true,
"license": "MIT",
"dependencies": {
"undici-types": "~6.19.2"
}
@ -5669,6 +5852,18 @@
"@types/send": "*"
}
},
"node_modules/@types/statuses": {
"version": "2.0.5",
"resolved": "https://registry.npmjs.org/@types/statuses/-/statuses-2.0.5.tgz",
"integrity": "sha512-jmIUGWrAiwu3dZpxntxieC+1n/5c3mjrImkmOSQ2NC5uP6cYO4aAZDdSmRcI5C1oiTmqlZGHC+/NmJrKogbP5A==",
"dev": true
},
"node_modules/@types/tough-cookie": {
"version": "4.0.5",
"resolved": "https://registry.npmjs.org/@types/tough-cookie/-/tough-cookie-4.0.5.tgz",
"integrity": "sha512-/Ad8+nIOV7Rl++6f1BdKxFSMgmoqEoYbHRpPcx3JEfv8VRsQe9Z4mCXeJBzxs7mbHY/XOZZuXlRNfhpVPbs6ZA==",
"dev": true
},
"node_modules/@types/unist": {
"version": "3.0.3",
"resolved": "https://registry.npmjs.org/@types/unist/-/unist-3.0.3.tgz",
@ -5681,6 +5876,12 @@
"integrity": "sha512-jg+97EGIcY9AGHJJRaaPVgetKDsrTgbRjQ5Msgjh/DQKEFl0DtyRr/VCOyD1T2R1MNeWPK/u7JoGhlDZnKBAfA==",
"dev": true
},
"node_modules/@types/wrap-ansi": {
"version": "3.0.0",
"resolved": "https://registry.npmjs.org/@types/wrap-ansi/-/wrap-ansi-3.0.0.tgz",
"integrity": "sha512-ltIpx+kM7g/MLRZfkbL7EsCEjfzCcScLpkg37eXEtx5kmrAKBkTJwd1GIAjDSL8wTpM6Hzn5YO4pSb91BEwu1g==",
"dev": true
},
"node_modules/@typescript-eslint/eslint-plugin": {
"version": "7.18.0",
"resolved": "https://registry.npmjs.org/@typescript-eslint/eslint-plugin/-/eslint-plugin-7.18.0.tgz",
@ -6069,6 +6270,33 @@
"url": "https://github.com/sponsors/epoberezkin"
}
},
"node_modules/ansi-escapes": {
"version": "4.3.2",
"resolved": "https://registry.npmjs.org/ansi-escapes/-/ansi-escapes-4.3.2.tgz",
"integrity": "sha512-gKXj5ALrKWQLsYG9jlTRmR/xKluxHV+Z9QEwNIgCfM1/uwPMCuzVVnh5mwTd+OuBZcwSIMbqssNWRm1lE51QaQ==",
"dev": true,
"dependencies": {
"type-fest": "^0.21.3"
},
"engines": {
"node": ">=8"
},
"funding": {
"url": "https://github.com/sponsors/sindresorhus"
}
},
"node_modules/ansi-escapes/node_modules/type-fest": {
"version": "0.21.3",
"resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.21.3.tgz",
"integrity": "sha512-t0rzBq87m3fVcduHDUFhKmyyX+9eo6WQjZvf51Ea/M0Q7+T374Jp1aUiyUl0GKxp8M/OETVHSDvmkyPgvX+X2w==",
"dev": true,
"engines": {
"node": ">=10"
},
"funding": {
"url": "https://github.com/sponsors/sindresorhus"
}
},
"node_modules/ansi-regex": {
"version": "5.0.1",
"resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz",
@ -6754,6 +6982,66 @@
"url": "https://github.com/sponsors/sindresorhus"
}
},
"node_modules/cli-width": {
"version": "4.1.0",
"resolved": "https://registry.npmjs.org/cli-width/-/cli-width-4.1.0.tgz",
"integrity": "sha512-ouuZd4/dm2Sw5Gmqy6bGyNNNe1qt9RpmxveLSO7KcgsTnU7RXfsw+/bukWGo1abgBiMAic068rclZsO4IWmmxQ==",
"dev": true,
"engines": {
"node": ">= 12"
}
},
"node_modules/cliui": {
"version": "8.0.1",
"resolved": "https://registry.npmjs.org/cliui/-/cliui-8.0.1.tgz",
"integrity": "sha512-BSeNnyus75C4//NQ9gQt1/csTXyo/8Sb+afLAkzAptFuMsod9HFokGNudZpi/oQV73hnVK+sR+5PVRMd+Dr7YQ==",
"dev": true,
"dependencies": {
"string-width": "^4.2.0",
"strip-ansi": "^6.0.1",
"wrap-ansi": "^7.0.0"
},
"engines": {
"node": ">=12"
}
},
"node_modules/cliui/node_modules/emoji-regex": {
"version": "8.0.0",
"resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz",
"integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==",
"dev": true
},
"node_modules/cliui/node_modules/string-width": {
"version": "4.2.3",
"resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz",
"integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==",
"dev": true,
"dependencies": {
"emoji-regex": "^8.0.0",
"is-fullwidth-code-point": "^3.0.0",
"strip-ansi": "^6.0.1"
},
"engines": {
"node": ">=8"
}
},
"node_modules/cliui/node_modules/wrap-ansi": {
"version": "7.0.0",
"resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-7.0.0.tgz",
"integrity": "sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==",
"dev": true,
"dependencies": {
"ansi-styles": "^4.0.0",
"string-width": "^4.1.0",
"strip-ansi": "^6.0.0"
},
"engines": {
"node": ">=10"
},
"funding": {
"url": "https://github.com/chalk/wrap-ansi?sponsor=1"
}
},
"node_modules/clone": {
"version": "1.0.4",
"resolved": "https://registry.npmjs.org/clone/-/clone-1.0.4.tgz",
@ -8235,6 +8523,15 @@
"node": ">=6.9.0"
}
},
"node_modules/get-caller-file": {
"version": "2.0.5",
"resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-2.0.5.tgz",
"integrity": "sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==",
"dev": true,
"engines": {
"node": "6.* || 8.* || >= 10.*"
}
},
"node_modules/get-func-name": {
"version": "2.0.2",
"resolved": "https://registry.npmjs.org/get-func-name/-/get-func-name-2.0.2.tgz",
@ -8457,6 +8754,15 @@
"dev": true,
"license": "MIT"
},
"node_modules/graphql": {
"version": "16.9.0",
"resolved": "https://registry.npmjs.org/graphql/-/graphql-16.9.0.tgz",
"integrity": "sha512-GGTKBX4SD7Wdb8mqeDLni2oaRGYQWjWHGKPQ24ZMnUtKfcsVoiv4uX8+LJr1K6U5VW2Lu1BwJnj7uiori0YtRw==",
"dev": true,
"engines": {
"node": "^12.22.0 || ^14.16.0 || ^16.0.0 || >=17.0.0"
}
},
"node_modules/has-flag": {
"version": "4.0.0",
"resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz",
@ -8569,6 +8875,12 @@
"url": "https://opencollective.com/unified"
}
},
"node_modules/headers-polyfill": {
"version": "4.0.3",
"resolved": "https://registry.npmjs.org/headers-polyfill/-/headers-polyfill-4.0.3.tgz",
"integrity": "sha512-IScLbePpkvO846sIwOtOTDjutRMWdXdJmXdMvk6gCBHxFO8d+QKOQedyZSxFTTFYRSmlgSTDtXqqq4pcenBXLQ==",
"dev": true
},
"node_modules/html-tags": {
"version": "3.3.1",
"resolved": "https://registry.npmjs.org/html-tags/-/html-tags-3.3.1.tgz",
@ -8850,6 +9162,12 @@
"node": ">=8"
}
},
"node_modules/is-node-process": {
"version": "1.2.0",
"resolved": "https://registry.npmjs.org/is-node-process/-/is-node-process-1.2.0.tgz",
"integrity": "sha512-Vg4o6/fqPxIjtxgUH5QLJhwZ7gW5diGCVlXpuUfELC62CuxM1iHcRe51f2W1FDy04Ai4KJkagKjx3XaqyfRKXw==",
"dev": true
},
"node_modules/is-number": {
"version": "7.0.0",
"resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz",
@ -9546,6 +9864,76 @@
"dev": true,
"license": "MIT"
},
"node_modules/msw": {
"version": "2.4.4",
"resolved": "https://registry.npmjs.org/msw/-/msw-2.4.4.tgz",
"integrity": "sha512-iuM0qGs4YmgYCLH+xqb07w2e/e4fYmsx3+WHVlIOUA34TW1sw+wRpNmOlXnLDkw/T7233Jnm6t+aNf4v2E3e2Q==",
"dev": true,
"hasInstallScript": true,
"dependencies": {
"@bundled-es-modules/cookie": "^2.0.0",
"@bundled-es-modules/statuses": "^1.0.1",
"@bundled-es-modules/tough-cookie": "^0.1.6",
"@inquirer/confirm": "^3.0.0",
"@mswjs/interceptors": "^0.35.0",
"@open-draft/until": "^2.1.0",
"@types/cookie": "^0.6.0",
"@types/statuses": "^2.0.4",
"chalk": "^4.1.2",
"graphql": "^16.8.1",
"headers-polyfill": "^4.0.2",
"is-node-process": "^1.2.0",
"outvariant": "^1.4.2",
"path-to-regexp": "^6.2.0",
"strict-event-emitter": "^0.5.1",
"type-fest": "^4.9.0",
"yargs": "^17.7.2"
},
"bin": {
"msw": "cli/index.js"
},
"engines": {
"node": ">=18"
},
"funding": {
"url": "https://github.com/sponsors/mswjs"
},
"peerDependencies": {
"typescript": ">= 4.8.x"
},
"peerDependenciesMeta": {
"typescript": {
"optional": true
}
}
},
"node_modules/msw/node_modules/path-to-regexp": {
"version": "6.2.2",
"resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-6.2.2.tgz",
"integrity": "sha512-GQX3SSMokngb36+whdpRXE+3f9V8UzyAorlYvOGx87ufGHehNTn5lCxrKtLyZ4Yl/wEKnNnr98ZzOwwDZV5ogw==",
"dev": true
},
"node_modules/msw/node_modules/type-fest": {
"version": "4.26.1",
"resolved": "https://registry.npmjs.org/type-fest/-/type-fest-4.26.1.tgz",
"integrity": "sha512-yOGpmOAL7CkKe/91I5O3gPICmJNLJ1G4zFYVAsRHg7M64biSnPtRj0WNQt++bRkjYOqjWXrhnUw1utzmVErAdg==",
"dev": true,
"engines": {
"node": ">=16"
},
"funding": {
"url": "https://github.com/sponsors/sindresorhus"
}
},
"node_modules/mute-stream": {
"version": "1.0.0",
"resolved": "https://registry.npmjs.org/mute-stream/-/mute-stream-1.0.0.tgz",
"integrity": "sha512-avsJQhyd+680gKXyG/sQc0nXaC6rBkPOfyHYcFb9+hdkqQkR9bdnkJ0AMZhke0oesPqIO+mFFJ+IdBc7mst4IA==",
"dev": true,
"engines": {
"node": "^14.17.0 || ^16.13.0 || >=18.0.0"
}
},
"node_modules/mz": {
"version": "2.7.0",
"resolved": "https://registry.npmjs.org/mz/-/mz-2.7.0.tgz",
@ -9844,6 +10232,12 @@
"url": "https://github.com/sponsors/sindresorhus"
}
},
"node_modules/outvariant": {
"version": "1.4.3",
"resolved": "https://registry.npmjs.org/outvariant/-/outvariant-1.4.3.tgz",
"integrity": "sha512-+Sl2UErvtsoajRDKCE5/dBz4DIvHXQQnAxtQTF04OJxY0+DyZXSo5P5Bb7XYWOh81syohlYL24hbDwxedPUJCA==",
"dev": true
},
"node_modules/p-limit": {
"version": "3.1.0",
"resolved": "https://registry.npmjs.org/p-limit/-/p-limit-3.1.0.tgz",
@ -10491,6 +10885,12 @@
"resolved": "https://registry.npmjs.org/proxy-from-env/-/proxy-from-env-1.1.0.tgz",
"integrity": "sha512-D+zkORCbA9f1tdWRK0RaCR3GPv50cMxcrz4X8k5LTSUD1Dkw47mKJEZQNunItRTkWwgtaUSo1RVFRIG9ZXiFYg=="
},
"node_modules/psl": {
"version": "1.9.0",
"resolved": "https://registry.npmjs.org/psl/-/psl-1.9.0.tgz",
"integrity": "sha512-E/ZsdU4HLs/68gYzgGTkMicWTLPdAftJLfJFlLUAAKZGkStNU72sZjT66SnMDVOfOWY/YAoiD7Jxa9iHvngcag==",
"dev": true
},
"node_modules/punycode": {
"version": "2.3.1",
"resolved": "https://registry.npmjs.org/punycode/-/punycode-2.3.1.tgz",
@ -10516,6 +10916,12 @@
"url": "https://github.com/sponsors/ljharb"
}
},
"node_modules/querystringify": {
"version": "2.2.0",
"resolved": "https://registry.npmjs.org/querystringify/-/querystringify-2.2.0.tgz",
"integrity": "sha512-FIqgj2EUvTa7R50u0rGsyTftzjYmv/a3hO345bZNrqabNqjtgiDMgmo4mkUjd+nzU5oF3dClKqFIPUKybUyqoQ==",
"dev": true
},
"node_modules/queue-microtask": {
"version": "1.2.3",
"resolved": "https://registry.npmjs.org/queue-microtask/-/queue-microtask-1.2.3.tgz",
@ -11048,6 +11454,15 @@
"url": "https://opencollective.com/unified"
}
},
"node_modules/require-directory": {
"version": "2.1.1",
"resolved": "https://registry.npmjs.org/require-directory/-/require-directory-2.1.1.tgz",
"integrity": "sha512-fGxEI7+wsG9xrvdjsrlmL22OMTTiHRwAMroiEeMgq8gzoLC/PQr7RsRDSTLUg/bZAZtF+TVIkHc6/4RIKrui+Q==",
"dev": true,
"engines": {
"node": ">=0.10.0"
}
},
"node_modules/requireindex": {
"version": "1.2.0",
"resolved": "https://registry.npmjs.org/requireindex/-/requireindex-1.2.0.tgz",
@ -11057,6 +11472,12 @@
"node": ">=0.10.5"
}
},
"node_modules/requires-port": {
"version": "1.0.0",
"resolved": "https://registry.npmjs.org/requires-port/-/requires-port-1.0.0.tgz",
"integrity": "sha512-KigOCHcocU3XODJxsu8i/j8T9tzT4adHiecwORRQ0ZZFcp7ahwXuRU1m+yuO90C5ZUyGeGfocHDI14M3L3yDAQ==",
"dev": true
},
"node_modules/resolve": {
"version": "1.22.8",
"resolved": "https://registry.npmjs.org/resolve/-/resolve-1.22.8.tgz",
@ -11714,6 +12135,12 @@
"node": ">=6"
}
},
"node_modules/strict-event-emitter": {
"version": "0.5.1",
"resolved": "https://registry.npmjs.org/strict-event-emitter/-/strict-event-emitter-0.5.1.tgz",
"integrity": "sha512-vMgjE/GGEPEFnhFub6pa4FmJBRBVOLpIII2hvCZ8Kzb7K0hlHo7mQv6xYrBvCL2LtAIBwFUK8wvuJgTVSQ5MFQ==",
"dev": true
},
"node_modules/string_decoder": {
"version": "1.3.0",
"resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.3.0.tgz",
@ -12193,6 +12620,30 @@
"node": ">=0.6"
}
},
"node_modules/tough-cookie": {
"version": "4.1.4",
"resolved": "https://registry.npmjs.org/tough-cookie/-/tough-cookie-4.1.4.tgz",
"integrity": "sha512-Loo5UUvLD9ScZ6jh8beX1T6sO1w2/MpCRpEP7V280GKMVUQ0Jzar2U3UJPsrdbziLEMMhu3Ujnq//rhiFuIeag==",
"dev": true,
"dependencies": {
"psl": "^1.1.33",
"punycode": "^2.1.1",
"universalify": "^0.2.0",
"url-parse": "^1.5.3"
},
"engines": {
"node": ">=6"
}
},
"node_modules/tough-cookie/node_modules/universalify": {
"version": "0.2.0",
"resolved": "https://registry.npmjs.org/universalify/-/universalify-0.2.0.tgz",
"integrity": "sha512-CJ1QgKmNg3CwvAv/kOFmtnEN05f0D/cn9QntgNOQlQF9dgvVTHj3t+8JPdjqawCHk7V/KA+fbUqzZ9XWhcqPUg==",
"dev": true,
"engines": {
"node": ">= 4.0.0"
}
},
"node_modules/ts-api-utils": {
"version": "1.3.0",
"resolved": "https://registry.npmjs.org/ts-api-utils/-/ts-api-utils-1.3.0.tgz",
@ -12540,6 +12991,16 @@
"punycode": "^2.1.0"
}
},
"node_modules/url-parse": {
"version": "1.5.10",
"resolved": "https://registry.npmjs.org/url-parse/-/url-parse-1.5.10.tgz",
"integrity": "sha512-WypcfiRhfeUP9vvF0j6rw0J3hrWrw6iZv3+22h6iRMJ/8z1Tj6XfLP4DsUix5MhMPnXpiHDoKyoZ/bdCkwBCiQ==",
"dev": true,
"dependencies": {
"querystringify": "^2.1.1",
"requires-port": "^1.0.0"
}
},
"node_modules/use-callback-ref": {
"version": "1.3.2",
"resolved": "https://registry.npmjs.org/use-callback-ref/-/use-callback-ref-1.3.2.tgz",
@ -13073,6 +13534,15 @@
}
}
},
"node_modules/y18n": {
"version": "5.0.8",
"resolved": "https://registry.npmjs.org/y18n/-/y18n-5.0.8.tgz",
"integrity": "sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA==",
"dev": true,
"engines": {
"node": ">=10"
}
},
"node_modules/yallist": {
"version": "4.0.0",
"resolved": "https://registry.npmjs.org/yallist/-/yallist-4.0.0.tgz",
@ -13091,6 +13561,53 @@
"node": ">= 14"
}
},
"node_modules/yargs": {
"version": "17.7.2",
"resolved": "https://registry.npmjs.org/yargs/-/yargs-17.7.2.tgz",
"integrity": "sha512-7dSzzRQ++CKnNI/krKnYRV7JKKPUXMEh61soaHKg9mrWEhzFWhFnxPxGl+69cD1Ou63C13NUPCnmIcrvqCuM6w==",
"dev": true,
"dependencies": {
"cliui": "^8.0.1",
"escalade": "^3.1.1",
"get-caller-file": "^2.0.5",
"require-directory": "^2.1.1",
"string-width": "^4.2.3",
"y18n": "^5.0.5",
"yargs-parser": "^21.1.1"
},
"engines": {
"node": ">=12"
}
},
"node_modules/yargs-parser": {
"version": "21.1.1",
"resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-21.1.1.tgz",
"integrity": "sha512-tVpsJW7DdjecAiFpbIB1e3qxIQsE6NoPc5/eTdrbbIC4h0LVsWhnoa3g+m2HclBIujHzsxZ4VJVA+GUuc2/LBw==",
"dev": true,
"engines": {
"node": ">=12"
}
},
"node_modules/yargs/node_modules/emoji-regex": {
"version": "8.0.0",
"resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz",
"integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==",
"dev": true
},
"node_modules/yargs/node_modules/string-width": {
"version": "4.2.3",
"resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz",
"integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==",
"dev": true,
"dependencies": {
"emoji-regex": "^8.0.0",
"is-fullwidth-code-point": "^3.0.0",
"strip-ansi": "^6.0.1"
},
"engines": {
"node": ">=8"
}
},
"node_modules/yocto-queue": {
"version": "0.1.0",
"resolved": "https://registry.npmjs.org/yocto-queue/-/yocto-queue-0.1.0.tgz",
@ -13104,6 +13621,18 @@
"url": "https://github.com/sponsors/sindresorhus"
}
},
"node_modules/yoctocolors-cjs": {
"version": "2.1.2",
"resolved": "https://registry.npmjs.org/yoctocolors-cjs/-/yoctocolors-cjs-2.1.2.tgz",
"integrity": "sha512-cYVsTjKl8b+FrnidjibDWskAv7UKOfcwaVZdp/it9n1s9fU3IkgDbhdIRKCW4JDsAlECJY0ytoVPT3sK6kideA==",
"dev": true,
"engines": {
"node": ">=18"
},
"funding": {
"url": "https://github.com/sponsors/sindresorhus"
}
},
"node_modules/zod": {
"version": "3.23.8",
"resolved": "https://registry.npmjs.org/zod/-/zod-3.23.8.tgz",

View File

@ -5,6 +5,7 @@
"type": "module",
"scripts": {
"dev": "vite",
"dev:no-mock": "VITE_MOCKING_ENABLED=false vite",
"build": "tsc -b && vite build",
"lint": "eslint . --ext ts,tsx --report-unused-disable-directives --max-warnings 0",
"preview": "vite preview",
@ -60,6 +61,7 @@
"eslint-plugin-react-hooks": "^4.6.2",
"eslint-plugin-react-refresh": "^0.4.7",
"eslint-plugin-storybook": "^0.8.0",
"msw": "^2.4.4",
"postcss": "^8.4.41",
"prettier": "^3.3.3",
"prettier-plugin-tailwindcss": "^0.6.6",
@ -70,5 +72,10 @@
"vite-plugin-svgr": "^4.2.0",
"vite-tsconfig-paths": "^5.0.1",
"vitest": "^2.0.5"
},
"msw": {
"workerDirectory": [
"public"
]
}
}

View File

@ -0,0 +1,284 @@
/* eslint-disable */
/* tslint:disable */
/**
* Mock Service Worker.
* @see https://github.com/mswjs/msw
* - Please do NOT modify this file.
* - Please do NOT serve this file on production.
*/
const PACKAGE_VERSION = '2.4.4'
const INTEGRITY_CHECKSUM = '26357c79639bfa20d64c0efca2a87423'
const IS_MOCKED_RESPONSE = Symbol('isMockedResponse')
const activeClientIds = new Set()
self.addEventListener('install', function () {
self.skipWaiting()
})
self.addEventListener('activate', function (event) {
event.waitUntil(self.clients.claim())
})
self.addEventListener('message', async function (event) {
const clientId = event.source.id
if (!clientId || !self.clients) {
return
}
const client = await self.clients.get(clientId)
if (!client) {
return
}
const allClients = await self.clients.matchAll({
type: 'window',
})
switch (event.data) {
case 'KEEPALIVE_REQUEST': {
sendToClient(client, {
type: 'KEEPALIVE_RESPONSE',
})
break
}
case 'INTEGRITY_CHECK_REQUEST': {
sendToClient(client, {
type: 'INTEGRITY_CHECK_RESPONSE',
payload: {
packageVersion: PACKAGE_VERSION,
checksum: INTEGRITY_CHECKSUM,
},
})
break
}
case 'MOCK_ACTIVATE': {
activeClientIds.add(clientId)
sendToClient(client, {
type: 'MOCKING_ENABLED',
payload: true,
})
break
}
case 'MOCK_DEACTIVATE': {
activeClientIds.delete(clientId)
break
}
case 'CLIENT_CLOSED': {
activeClientIds.delete(clientId)
const remainingClients = allClients.filter((client) => {
return client.id !== clientId
})
// Unregister itself when there are no more clients
if (remainingClients.length === 0) {
self.registration.unregister()
}
break
}
}
})
self.addEventListener('fetch', function (event) {
const { request } = event
// Bypass navigation requests.
if (request.mode === 'navigate') {
return
}
// Opening the DevTools triggers the "only-if-cached" request
// that cannot be handled by the worker. Bypass such requests.
if (request.cache === 'only-if-cached' && request.mode !== 'same-origin') {
return
}
// Bypass all requests when there are no active clients.
// Prevents the self-unregistered worked from handling requests
// after it's been deleted (still remains active until the next reload).
if (activeClientIds.size === 0) {
return
}
// Generate unique request ID.
const requestId = crypto.randomUUID()
event.respondWith(handleRequest(event, requestId))
})
async function handleRequest(event, requestId) {
const client = await resolveMainClient(event)
const response = await getResponse(event, client, requestId)
// Send back the response clone for the "response:*" life-cycle events.
// Ensure MSW is active and ready to handle the message, otherwise
// this message will pend indefinitely.
if (client && activeClientIds.has(client.id)) {
;(async function () {
const responseClone = response.clone()
sendToClient(
client,
{
type: 'RESPONSE',
payload: {
requestId,
isMockedResponse: IS_MOCKED_RESPONSE in response,
type: responseClone.type,
status: responseClone.status,
statusText: responseClone.statusText,
body: responseClone.body,
headers: Object.fromEntries(responseClone.headers.entries()),
},
},
[responseClone.body],
)
})()
}
return response
}
// Resolve the main client for the given event.
// Client that issues a request doesn't necessarily equal the client
// that registered the worker. It's with the latter the worker should
// communicate with during the response resolving phase.
async function resolveMainClient(event) {
const client = await self.clients.get(event.clientId)
if (client?.frameType === 'top-level') {
return client
}
const allClients = await self.clients.matchAll({
type: 'window',
})
return allClients
.filter((client) => {
// Get only those clients that are currently visible.
return client.visibilityState === 'visible'
})
.find((client) => {
// Find the client ID that's recorded in the
// set of clients that have registered the worker.
return activeClientIds.has(client.id)
})
}
async function getResponse(event, client, requestId) {
const { request } = event
// Clone the request because it might've been already used
// (i.e. its body has been read and sent to the client).
const requestClone = request.clone()
function passthrough() {
const headers = Object.fromEntries(requestClone.headers.entries())
// Remove internal MSW request header so the passthrough request
// complies with any potential CORS preflight checks on the server.
// Some servers forbid unknown request headers.
delete headers['x-msw-intention']
return fetch(requestClone, { headers })
}
// Bypass mocking when the client is not active.
if (!client) {
return passthrough()
}
// Bypass initial page load requests (i.e. static assets).
// The absence of the immediate/parent client in the map of the active clients
// means that MSW hasn't dispatched the "MOCK_ACTIVATE" event yet
// and is not ready to handle requests.
if (!activeClientIds.has(client.id)) {
return passthrough()
}
// Notify the client that a request has been intercepted.
const requestBuffer = await request.arrayBuffer()
const clientMessage = await sendToClient(
client,
{
type: 'REQUEST',
payload: {
id: requestId,
url: request.url,
mode: request.mode,
method: request.method,
headers: Object.fromEntries(request.headers.entries()),
cache: request.cache,
credentials: request.credentials,
destination: request.destination,
integrity: request.integrity,
redirect: request.redirect,
referrer: request.referrer,
referrerPolicy: request.referrerPolicy,
body: requestBuffer,
keepalive: request.keepalive,
},
},
[requestBuffer],
)
switch (clientMessage.type) {
case 'MOCK_RESPONSE': {
return respondWithMock(clientMessage.data)
}
case 'PASSTHROUGH': {
return passthrough()
}
}
return passthrough()
}
function sendToClient(client, message, transferrables = []) {
return new Promise((resolve, reject) => {
const channel = new MessageChannel()
channel.port1.onmessage = (event) => {
if (event.data && event.data.error) {
return reject(event.data.error)
}
resolve(event.data)
}
client.postMessage(
message,
[channel.port2].concat(transferrables.filter(Boolean)),
)
})
}
async function respondWithMock(response) {
// Setting response status code to 0 is a no-op.
// However, when responding with a "Response.error()", the produced Response
// instance will have status code set to 0. Since it's not possible to create
// a Response instance with status code 0, handle that use-case separately.
if (response.status === 0) {
return Response.error()
}
const mockedResponse = new Response(response.body, response)
Reflect.defineProperty(mockedResponse, IS_MOCKED_RESPONSE, {
value: true,
enumerable: true,
})
return mockedResponse
}

View File

@ -0,0 +1,22 @@
import api from '@/api/axiosConfig';
import { AxiosError } from 'axios';
export const reissueTokenApi = () => {
return api.post('/api/auth/reissue', null, { withCredentials: true }).then((response) => response.data);
};
export const fetchProfileApi = async () => {
try {
const response = await api.get('/api/auth/profile', {
withCredentials: true,
});
return response.data;
} catch (error) {
if (error instanceof AxiosError) {
console.error('사용자 정보 가져오기 실패:', error.response?.data?.message || '알 수 없는 오류');
} else {
console.error('알 수 없는 오류가 발생했습니다.');
}
throw error;
}
};

View File

@ -0,0 +1,102 @@
import axios, { AxiosError, AxiosResponse, InternalAxiosRequestConfig } from 'axios';
import useAuthStore from '@/stores/useAuthStore';
const baseURL = 'https://j11s002.p.ssafy.io';
const api = axios.create({
baseURL,
withCredentials: true,
});
let isTokenRefreshing = false;
type FailedRequest = {
resolve: (value?: string | undefined) => void;
reject: (reason?: unknown) => void;
};
let failedQueue: FailedRequest[] = [];
const processQueue = (error: Error | null, token: string | undefined = undefined): void => {
failedQueue.forEach((prom) => {
if (error) {
prom.reject(error);
} else {
prom.resolve(token);
}
});
failedQueue = [];
};
api.interceptors.request.use((config: InternalAxiosRequestConfig) => {
const token = localStorage.getItem('accessToken');
if (token && config.headers) {
config.headers.Authorization = `Bearer ${token}`;
}
return config;
});
api.interceptors.response.use(
(response: AxiosResponse) => response,
async (error: AxiosError) => {
const originalRequest = error.config as InternalAxiosRequestConfig & { _retry?: boolean };
if (error.response?.status === 401 && !originalRequest._retry) {
if (isTokenRefreshing) {
return new Promise<string | undefined>((resolve, reject) => {
failedQueue.push({ resolve, reject });
})
.then((token) => {
if (token && originalRequest.headers) {
originalRequest.headers.Authorization = `Bearer ${token}`;
}
return api(originalRequest);
})
.catch((err) => Promise.reject(err));
}
originalRequest._retry = true;
isTokenRefreshing = true;
return api
.post('/api/auth/reissue', null, { withCredentials: true })
.then((response) => {
const newAccessToken = response.data?.data?.accessToken;
if (!newAccessToken) {
throw new Error('Invalid token reissue response');
}
useAuthStore.getState().setLoggedIn(true, newAccessToken);
localStorage.setItem('accessToken', newAccessToken);
processQueue(null, newAccessToken);
if (originalRequest.headers) {
originalRequest.headers.Authorization = `Bearer ${newAccessToken}`;
}
return api(originalRequest);
})
.catch((reissueError: Error) => {
processQueue(reissueError, undefined);
console.error('토큰 재발급 실패:', reissueError);
useAuthStore.getState().clearAuth();
window.location.href = '/';
return Promise.reject(reissueError);
})
.finally(() => {
isTokenRefreshing = false;
});
}
if (error.response?.status === 400) {
alert('잘못된 요청입니다. 다시 시도해 주세요.');
} else if (error.response?.status === 403) {
alert('권한이 없습니다. 다시 로그인해 주세요.');
useAuthStore.getState().clearAuth();
window.location.href = '/';
}
return Promise.reject(error);
}
);
export default api;

View File

@ -0,0 +1,153 @@
import api from '@/api/axiosConfig';
import { AxiosError } from 'axios';
export const getProjectApi = async (projectId: number, memberId: number) => {
try {
const response = await api.get(`/api/projects/${projectId}`, {
params: {
memberId,
},
});
return response.data;
} catch (error) {
if (error instanceof AxiosError) {
console.error('프로젝트 조회 실패:', error.response?.data?.message || '알 수 없는 오류');
} else {
console.error('알 수 없는 오류가 발생했습니다.');
}
throw error;
}
};
export const updateProjectApi = async (
projectId: number,
memberId: number,
data: { title: string; projectType: string }
) => {
try {
const response = await api.put(`/api/projects/${projectId}`, data, {
params: {
memberId,
},
});
return response.data;
} catch (error) {
if (error instanceof AxiosError) {
console.error('프로젝트 수정 실패:', error.response?.data?.message || '알 수 없는 오류');
} else {
console.error('알 수 없는 오류가 발생했습니다.');
}
throw error;
}
};
export const deleteProjectApi = async (projectId: number, memberId: number) => {
try {
const response = await api.delete(`/api/projects/${projectId}`, {
params: {
memberId,
},
});
return response.data;
} catch (error) {
if (error instanceof AxiosError) {
console.error('프로젝트 삭제 실패:', error.response?.data?.message || '알 수 없는 오류');
} else {
console.error('알 수 없는 오류가 발생했습니다.');
}
throw error;
}
};
export const addProjectMemberApi = async (
projectId: number,
memberId: number,
newMemberId: number,
privilegeType: string
) => {
try {
const response = await api.post(
`/api/projects/${projectId}/members`,
{ memberId: newMemberId, privilegeType },
{
params: {
memberId,
},
}
);
return response.data;
} catch (error) {
if (error instanceof AxiosError) {
console.error('프로젝트 멤버 추가 실패:', error.response?.data?.message || '알 수 없는 오류');
} else {
console.error('알 수 없는 오류가 발생했습니다.');
}
throw error;
}
};
export const removeProjectMemberApi = async (projectId: number, memberId: number, targetMemberId: number) => {
try {
const response = await api.delete(`/api/projects/${projectId}/members`, {
params: {
memberId,
},
data: { memberId: targetMemberId },
});
return response.data;
} catch (error) {
if (error instanceof AxiosError) {
console.error('프로젝트 멤버 제거 실패:', error.response?.data?.message || '알 수 없는 오류');
} else {
console.error('알 수 없는 오류가 발생했습니다.');
}
throw error;
}
};
export const getAllProjectsApi = async (
workspaceId: number,
memberId: number,
lastProjectId?: number,
limit: number = 10
) => {
try {
const response = await api.get(`/api/workspaces/${workspaceId}/projects`, {
params: {
memberId,
lastProjectId,
limit,
},
});
return response.data;
} catch (error) {
if (error instanceof AxiosError) {
console.error('프로젝트 목록 조회 실패:', error.response?.data?.message || '알 수 없는 오류');
} else {
console.error('알 수 없는 오류가 발생했습니다.');
}
throw error;
}
};
export const createProjectApi = async (
workspaceId: number,
memberId: number,
data: { title: string; projectType: string }
) => {
try {
const response = await api.post(`/api/workspaces/${workspaceId}/projects`, data, {
params: {
memberId,
},
});
return response.data;
} catch (error) {
if (error instanceof AxiosError) {
console.error('프로젝트 생성 실패:', error.response?.data?.message || '알 수 없는 오류');
} else {
console.error('알 수 없는 오류가 발생했습니다.');
}
throw error;
}
};

View File

@ -0,0 +1,145 @@
import api from '@/api/axiosConfig';
import { AxiosError } from 'axios';
interface Workspace {
id: number;
memberId: string;
title: string;
content: string;
createdAt: string;
updatedAt: string;
}
interface GetAllWorkspacesResponse {
status: number;
code: number;
message: string;
data: {
workspaceResponses: Workspace[];
};
errors: Array<{
field: string;
code: string;
message: string;
objectName: string;
}>;
isSuccess: boolean;
}
export const getWorkspaceApi = async (workspaceId: number, memberId: number) => {
try {
const response = await api.get(`/api/workspaces/${workspaceId}`, {
params: { memberId },
});
return response.data;
} catch (error) {
if (error instanceof AxiosError) {
console.error('API 요청 실패:', error.response?.data?.message || '알 수 없는 오류');
} else {
console.error('알 수 없는 오류가 발생했습니다.');
}
throw error;
}
};
export const updateWorkspaceApi = async (
workspaceId: number,
memberId: number,
data: { title: string; content: string }
) => {
try {
const response = await api.put(`/api/workspaces/${workspaceId}`, data, {
params: { memberId },
});
return response.data;
} catch (error) {
if (error instanceof AxiosError) {
console.error('API 요청 실패:', error.response?.data?.message || '알 수 없는 오류');
} else {
console.error('알 수 없는 오류가 발생했습니다.');
}
throw error;
}
};
export const deleteWorkspaceApi = async (workspaceId: number, memberId: number) => {
try {
const response = await api.delete(`/api/workspaces/${workspaceId}`, {
params: { memberId },
});
return response.data;
} catch (error) {
if (error instanceof AxiosError) {
console.error('API 요청 실패:', error.response?.data?.message || '알 수 없는 오류');
} else {
console.error('알 수 없는 오류가 발생했습니다.');
}
throw error;
}
};
export const getAllWorkspacesApi = async (
memberId: number,
lastWorkspaceId?: number,
limit?: number
): Promise<GetAllWorkspacesResponse> => {
try {
const response = await api.get('/api/workspaces', {
params: { memberId, lastWorkspaceId, limit },
});
return response.data;
} catch (error) {
if (error instanceof AxiosError) {
console.error('API 요청 실패:', error.response?.data?.message || '알 수 없는 오류');
} else {
console.error('알 수 없는 오류가 발생했습니다.');
}
throw error;
}
};
export const createWorkspaceApi = async (memberId: number, data: { title: string; content: string }) => {
try {
const response = await api.post('/api/workspaces', data, {
params: { memberId },
});
return response.data;
} catch (error) {
if (error instanceof AxiosError) {
console.error('API 요청 실패:', error.response?.data?.message || '알 수 없는 오류');
} else {
console.error('알 수 없는 오류가 발생했습니다.');
}
throw error;
}
};
export const addWorkspaceMemberApi = async (workspaceId: number, memberId: number, newMemberId: number) => {
try {
const response = await api.post(`/api/workspaces/${workspaceId}/members/${newMemberId}`, null, {
params: { memberId },
});
return response.data;
} catch (error) {
if (error instanceof AxiosError) {
console.error('API 요청 실패:', error.response?.data?.message || '알 수 없는 오류');
} else {
console.error('알 수 없는 오류가 발생했습니다.');
}
throw error;
}
};
export const removeWorkspaceMemberApi = async (workspaceId: number, memberId: number, targetMemberId: number) => {
try {
const response = await api.delete(`/api/workspaces/${workspaceId}/members/${targetMemberId}`, {
params: { memberId },
});
return response.data;
} catch (error) {
if (error instanceof AxiosError) {
console.error('API 요청 실패:', error.response?.data?.message || '알 수 없는 오류');
} else {
console.error('알 수 없는 오류가 발생했습니다.');
}
throw error;
}
};

View File

@ -1,35 +0,0 @@
import '@/index.css';
import { Meta, StoryObj } from '@storybook/react';
import Home from '.';
import { useState } from 'react';
const HomeWrapper = ({ initialLoggedIn }: { initialLoggedIn: boolean }) => {
const [isLoggedIn, setIsLoggedIn] = useState(initialLoggedIn);
return (
<Home
isLoggedIn={isLoggedIn}
setIsLoggedIn={setIsLoggedIn}
/>
);
};
const meta: Meta<typeof Home> = {
title: 'Components/Home',
component: Home,
parameters: {
layout: 'fullscreen',
},
};
export default meta;
type Story = StoryObj<typeof Home>;
export const GoogleLogin: Story = {
render: () => <HomeWrapper initialLoggedIn={false} />,
};
export const SelectWorkspace: Story = {
render: () => <HomeWrapper initialLoggedIn={true} />,
};

View File

@ -1,39 +1,59 @@
import { useState } from 'react';
import { useRef } from 'react';
import { useNavigate } from 'react-router-dom';
import { Select, SelectTrigger, SelectContent, SelectItem, SelectValue, SelectGroup } from '../ui/select';
import GoogleLogo from '@/assets/icons/web_neutral_rd_ctn@1x.png';
import useAuthStore from '@/stores/useAuthStore';
import { Button } from '@/components/ui/button';
import { fetchProfileApi, reissueTokenApi } from '@/api/authApi';
interface HomeProps {
isLoggedIn?: boolean;
setIsLoggedIn?: (loggedIn: boolean) => void;
}
const DOMAIN = 'https://j11s002.p.ssafy.io';
export default function Home({ isLoggedIn = false, setIsLoggedIn }: HomeProps) {
const [, setSelectedWorkspace] = useState('');
const [loggedIn, setLoggedIn] = useState(isLoggedIn);
export default function Home() {
const navigate = useNavigate();
const { isLoggedIn, setLoggedIn, profile, setProfile } = useAuthStore();
const hasFetchedProfile = useRef(false);
const workspaces = [
{ id: 1, name: 'Workspace 1' },
{ id: 2, name: 'Workspace 2' },
{ id: 3, name: 'Workspace 3' },
];
if (!isLoggedIn && !profile.id && !hasFetchedProfile.current) {
const accessToken = localStorage.getItem('accessToken');
if (accessToken) {
setLoggedIn(true, accessToken);
fetchProfileApi()
.then((data) => {
if (data?.isSuccess && data.data) {
setProfile({
id: data.data.id,
nickname: data.data.nickname,
profileImage: data.data.profileImage,
});
hasFetchedProfile.current = true;
}
})
.catch((error) => {
alert('프로필을 가져오는 중 오류가 발생했습니다. 다시 시도해주세요.');
console.error('프로필 가져오기 실패:', error);
});
}
}
const handleGoogleSignIn = () => {
console.log('구글로 계속하기');
setLoggedIn(true);
if (setIsLoggedIn) {
setIsLoggedIn(true);
window.location.href = `${DOMAIN}/api/login/oauth2/authorization/google`;
};
const handleStart = () => {
navigate('/browse');
};
const handleReissueToken = async () => {
try {
const response = await reissueTokenApi();
console.log('토큰 재발급 성공:', response);
alert('토큰 재발급 성공! 새로운 액세스 토큰을 콘솔에서 확인하세요.');
} catch (error) {
console.error('토큰 재발급 실패:', error);
alert('토큰 재발급에 실패했습니다. 다시 시도해 주세요.');
}
};
const handleWorkspaceSelect = (value: string) => {
const selected = workspaces.find((workspace) => workspace.name === value);
if (selected) {
navigate(`/browse/${selected.id}`);
}
setSelectedWorkspace(value);
};
const isHidden = true;
return (
<div className="flex h-full flex-col items-center justify-center bg-gray-50 p-8">
@ -41,7 +61,7 @@ export default function Home({ isLoggedIn = false, setIsLoggedIn }: HomeProps) {
<h2 className="mb-4 text-2xl font-bold text-gray-900"> </h2>
<p className="mb-4 text-base text-gray-700">
(AI)
. .
.
</p>
<p className="mb-4 text-base text-gray-700">
.
@ -56,7 +76,7 @@ export default function Home({ isLoggedIn = false, setIsLoggedIn }: HomeProps) {
</p>
</div>
{!loggedIn ? (
{!isLoggedIn ? (
<button
onClick={handleGoogleSignIn}
className="mb-4 transition hover:opacity-90 focus:outline-none focus:ring-2 focus:ring-gray-300 active:opacity-80"
@ -68,23 +88,24 @@ export default function Home({ isLoggedIn = false, setIsLoggedIn }: HomeProps) {
/>
</button>
) : (
<Select onValueChange={handleWorkspaceSelect}>
<SelectTrigger className="mb-4 w-72">
<SelectValue placeholder="워크스페이스를 선택하세요" />
</SelectTrigger>
<SelectContent>
<SelectGroup>
{workspaces.map((workspace) => (
<SelectItem
key={workspace.id}
value={workspace.name}
>
{workspace.name}
</SelectItem>
))}
</SelectGroup>
</SelectContent>
</Select>
<>
<Button
variant="outlinePrimary"
size="lg"
onClick={handleStart}
>
</Button>
<Button
variant="outlinePrimary"
size="lg"
onClick={handleReissueToken}
className="mt-4"
style={{ display: isHidden ? 'none' : 'block' }}
>
</Button>
</>
)}
</div>
);

View File

@ -0,0 +1,43 @@
import { useEffect } from 'react';
import { useNavigate } from 'react-router-dom';
import useAuthStore from '@/stores/useAuthStore';
import { fetchProfileApi } from '@/api/authApi';
export default function OAuthCallback() {
const navigate = useNavigate();
const setLoggedIn = useAuthStore((state) => state.setLoggedIn);
useEffect(() => {
const queryParams = new URLSearchParams(window.location.search);
const accessToken = queryParams.get('accessToken');
if (accessToken) {
setLoggedIn(true, accessToken);
localStorage.setItem('accessToken', accessToken);
fetchProfileApi()
.then((data) => {
if (data?.isSuccess && data.data) {
const profileData = {
id: data.data.id,
nickname: data.data.nickname,
profileImage: data.data.profileImage,
};
useAuthStore.getState().setProfile(profileData);
navigate('/browse');
} else {
throw new Error('프로필 데이터를 가져올 수 없습니다.');
}
})
.catch((error) => {
alert('프로필을 가져오는 중 오류가 발생했습니다. 다시 로그인해주세요.');
console.error('프로필 가져오기 실패:', error);
navigate('/');
});
} else {
navigate('/');
}
}, [navigate, setLoggedIn]);
return <p> ...</p>;
}

View File

@ -1,25 +1,47 @@
import * as React from 'react';
import ProjectCreateForm, { ProjectCreateFormValues } from './ProjectCreateForm';
import XIcon from '@/assets/icons/x.svg?react';
import { Dialog, DialogContent, DialogHeader, DialogTrigger } from '../ui/dialogCustom';
import { Button } from '@/components/ui/button';
import { Plus } from 'lucide-react';
export default function ProjectCreateModal({
onClose,
onSubmit,
}: {
onClose: () => void;
onSubmit: (data: ProjectCreateFormValues) => void;
onSubmit: (data: { title: string; labelType: 'Classification' | 'Detection' | 'Segmentation' }) => void;
}) {
const [isOpen, setIsOpen] = React.useState(false);
const handleOpen = () => setIsOpen(true);
const handleClose = () => setIsOpen(false);
return (
<div className="flex w-[610px] flex-col gap-10 rounded-3xl border px-10 py-5 shadow-lg">
<header className="flex gap-5">
<h1 className="small-title w-full"> </h1>
<button
className="flex h-8 w-8 items-center justify-center"
onClick={onClose}
<Dialog
open={isOpen}
onOpenChange={setIsOpen}
>
<DialogTrigger asChild>
<Button
variant="outlinePrimary"
className="mt-4 flex items-center gap-2"
onClick={handleOpen}
>
<XIcon className="stroke-gray-900" />
</button>
</header>
<ProjectCreateForm onSubmit={onSubmit} />
</div>
<Plus size={16} />
<span> </span>
</Button>
</DialogTrigger>
<DialogContent>
<DialogHeader title="새 프로젝트" />
<ProjectCreateForm
onSubmit={(data: ProjectCreateFormValues) => {
const formattedData = {
title: data.projectName,
labelType: data.labelType,
};
onSubmit(formattedData);
handleClose();
}}
/>
</DialogContent>
</Dialog>
);
}

View File

@ -8,9 +8,6 @@ export default {
export const Default = () => (
<WorkSpaceCreateModal
onClose={() => {
console.log('close');
}}
onSubmit={(data) => {
console.log(data);
}}

View File

@ -1,25 +1,44 @@
import * as React from 'react';
import WorkSpaceCreateForm, { WorkSpaceCreateFormValues } from './WorkSpaceCreateForm';
import XIcon from '@/assets/icons/x.svg?react';
import { Dialog, DialogContent, DialogHeader, DialogTrigger } from '../ui/dialogCustom';
import { Plus } from 'lucide-react';
export default function WorkSpaceCreateModal({
onClose,
onSubmit,
}: {
onClose: () => void;
onSubmit: (data: WorkSpaceCreateFormValues) => void;
onSubmit: (data: { title: string; content: string }) => void;
}) {
const [isOpen, setIsOpen] = React.useState(false);
const handleOpen = () => setIsOpen(true);
const handleClose = () => setIsOpen(false);
return (
<div className="flex w-[610px] flex-col gap-10 rounded-3xl border px-10 py-5 shadow-lg">
<header className="flex gap-5">
<h1 className="small-title w-full"> </h1>
<Dialog
open={isOpen}
onOpenChange={setIsOpen}
>
<DialogTrigger asChild>
<button
className="flex h-8 w-8 items-center justify-center"
onClick={onClose}
className="flex items-center justify-center p-2"
onClick={handleOpen}
>
<XIcon className="stroke-gray-900" />
<Plus size={20} />
</button>
</header>
<WorkSpaceCreateForm onSubmit={onSubmit} />
</div>
</DialogTrigger>
<DialogContent>
<DialogHeader title="새 워크스페이스" />
<WorkSpaceCreateForm
onSubmit={(data: WorkSpaceCreateFormValues) => {
const formattedData = {
title: data.workspaceName,
content: data.workspaceDescription || '',
};
onSubmit(formattedData);
handleClose();
}}
/>
</DialogContent>
</Dialog>
);
}

View File

@ -1,71 +1,84 @@
import { useParams } from 'react-router-dom';
import ProjectCard from '@/components/ProjectCard';
import { Button } from '@/components/ui/button';
import { Workspace } from '@/types';
import { Plus, Smile } from 'lucide-react';
import { Dialog, DialogContent, DialogHeader, DialogTrigger } from '../ui/dialogCustom';
import WorkSpaceCreateForm from '../WorkSpaceCreateModal/WorkSpaceCreateForm';
import { Smile } from 'lucide-react';
import ProjectCreateModal from '../ProjectCreateModal';
import { useGetAllProjects, useCreateProject } from '@/hooks/useProjectHooks';
import useAuthStore from '@/stores/useAuthStore';
import { Key } from 'react';
export default function WorkspaceBrowseDetail() {
const { workspaceId } = useParams<{ workspaceId: string }>();
const numericWorkspaceId: number = Number(workspaceId);
const workspace: Workspace = !workspaceId
? {
id: 0,
name: '',
projects: [],
const numericWorkspaceId = Number(workspaceId);
const { profile } = useAuthStore();
const memberId = profile.id ?? 0;
const { data: projectsResponse, isLoading, isError, refetch } = useGetAllProjects(numericWorkspaceId || 0, memberId);
const createProject = useCreateProject();
const handleCreateProject = (data: { title: string; labelType: 'Classification' | 'Detection' | 'Segmentation' }) => {
createProject.mutate(
{
workspaceId: numericWorkspaceId,
memberId,
data: { title: data.title, projectType: data.labelType.toLowerCase() },
},
{
onSuccess: () => {
console.log('프로젝트가 성공적으로 생성되었습니다.');
refetch();
},
onError: (error) => {
console.error('프로젝트 생성 실패:', error);
console.log('Error details:', JSON.stringify(error, null, 2));
const errorMessage = error?.response?.data?.message || error.message || '알 수 없는 오류';
console.error('프로젝트 생성 실패:', errorMessage);
},
}
: {
id: numericWorkspaceId,
name: `workspace-${workspaceId}`,
projects: [
{ id: 1, name: 'project1', type: 'Detection', children: [] },
{ id: 2, name: 'project2', type: 'Detection', children: [] },
{ id: 3, name: 'project3', type: 'Detection', children: [] },
{ id: 4, name: 'project4', type: 'Detection', children: [] },
{ id: 5, name: 'project5', type: 'Detection', children: [] },
],
};
);
};
const projects = Array.isArray(projectsResponse?.data?.workspaceResponses)
? projectsResponse.data.workspaceResponses
: [];
if (isLoading) {
return <p>Loading projects...</p>;
}
if (isError || !workspaceId) {
return (
<div className="flex h-full w-full flex-col items-center justify-center">
<div className="flex flex-col items-center">
<Smile
size={48}
className="mb-2 text-gray-300"
/>
<div className="body text-gray-400">
{!workspaceId ? '작업할 워크스페이스를 선택하세요.' : '작업할 프로젝트가 없습니다.'}
</div>
</div>
</div>
);
}
return (
<div className="flex h-full w-full flex-col gap-8 px-6 py-4">
<div className="flex items-center justify-center">
<h1 className="small-title flex grow">{workspaceId ? workspace.name : ''}</h1>
<h1 className="small-title flex grow">{`Workspace-${workspaceId}`}</h1>
<div className="flex flex-col">
<div className="flex gap-3">
<Dialog>
<DialogTrigger asChild>
<Button
variant="outline"
onClick={() => {
console.log('새 프로젝트 생성 모달');
}}
>
<div className="body flex items-center gap-2">
<Plus size={16} />
<span> </span>
</div>
</Button>
</DialogTrigger>
<DialogContent>
<DialogHeader title="새 워크스페이스" />
<WorkSpaceCreateForm
onSubmit={(data) => {
console.log(data);
}}
/>
</DialogContent>
</Dialog>
<ProjectCreateModal onSubmit={handleCreateProject} />
</div>
</div>
</div>
{workspaceId ? (
{projects.length > 0 ? (
<div className="flex flex-wrap gap-6">
{workspace.projects.map((project) => (
{projects.map((project: { id: Key | null | undefined; title: string; projectType: string }) => (
<ProjectCard
key={project.id}
title={project.name}
description={project.type}
title={project.title}
description={project.projectType}
onClick={() => {
console.log('project id : ' + project.id);
}}
@ -73,13 +86,13 @@ export default function WorkspaceBrowseDetail() {
))}
</div>
) : (
<div className="flex w-full grow items-center justify-center">
<div className="flex h-full w-full flex-col items-center justify-center">
<div className="flex flex-col items-center">
<Smile
size={48}
className="mb-2 text-gray-300"
/>
<div className="body text-gray-400"> .</div>
<div className="body text-gray-400"> .</div>
</div>
</div>
)}

View File

@ -1,39 +1,54 @@
import { Suspense } from 'react';
import { NavLink, Outlet } from 'react-router-dom';
import { Suspense, useEffect } from 'react';
import { NavLink, Outlet, useNavigate } from 'react-router-dom';
import Header from '../Header';
import { Workspace } from '@/types';
import { Dialog, DialogContent, DialogHeader, DialogTrigger } from '../ui/dialogCustom';
import { Plus } from 'lucide-react';
import WorkSpaceCreateForm from '../WorkSpaceCreateModal/WorkSpaceCreateForm';
import { useGetAllWorkspaces, useCreateWorkspace } from '@/hooks/useWorkspaceHooks';
import useAuthStore from '@/stores/useAuthStore';
import WorkSpaceCreateModal from '../WorkSpaceCreateModal';
import { useQueryClient } from '@tanstack/react-query';
import { AxiosError } from 'axios';
export default function WorkspaceBrowseLayout() {
const workspaces: Workspace[] = [
{
id: 1,
name: 'workspace-1',
projects: [],
},
{
id: 2,
name: 'workspace-2',
projects: [],
},
{
id: 3,
name: 'workspace-3',
projects: [],
},
{
id: 4,
name: 'workspace-4',
projects: [],
},
{
id: 5,
name: 'workspace-5',
projects: [],
},
];
const { profile, isLoggedIn } = useAuthStore();
const memberId = profile.id;
const navigate = useNavigate();
const queryClient = useQueryClient();
useEffect(() => {
if (!isLoggedIn || !memberId) {
console.error('로그인되지 않았거나 유효한 멤버 ID가 없습니다.');
navigate('/');
}
}, [isLoggedIn, memberId, navigate]);
const { data: workspacesResponse, isLoading, isError } = useGetAllWorkspaces(memberId || 0);
const workspaces = workspacesResponse?.data?.workspaceResponses || [];
const createWorkspace = useCreateWorkspace();
const handleCreateWorkspace = (data: { title: string; content: string }) => {
if (!memberId) return;
createWorkspace.mutate(
{ memberId, data },
{
onSuccess: () => {
console.log('워크스페이스가 성공적으로 생성되었습니다.');
queryClient.invalidateQueries({ queryKey: ['workspaces'] });
},
onError: (error: AxiosError) => {
console.error('워크스페이스 생성 실패:', error.message);
},
}
);
};
if (isLoading) {
return <p>Loading workspaces...</p>;
}
if (isError) {
return <p>Error loading workspaces. Please try again later.</p>;
}
return (
<>
@ -43,21 +58,7 @@ export default function WorkspaceBrowseLayout() {
<div className="flex w-[280px] flex-col gap-4 border-r border-gray-200 bg-gray-100 px-6 py-4">
<div className="flex items-center justify-center gap-5">
<h1 className="heading mr-2.5 w-full"> </h1>
<Dialog>
<DialogTrigger asChild>
<button className="p-2">
<Plus size={20} />
</button>
</DialogTrigger>
<DialogContent>
<DialogHeader title="새 워크스페이스" />
<WorkSpaceCreateForm
onSubmit={(data) => {
console.log(data);
}}
/>
</DialogContent>
</Dialog>
<WorkSpaceCreateModal onSubmit={handleCreateWorkspace} />
</div>
{workspaces.length > 0 ? (
workspaces.map((workspace) => (
@ -66,7 +67,7 @@ export default function WorkspaceBrowseLayout() {
key={workspace.id}
className={({ isActive }) => (isActive ? 'body-strong' : 'body') + ' cursor-pointer'}
>
{workspace.name}
{workspace.title}
</NavLink>
))
) : (

View File

@ -0,0 +1,104 @@
import { useQuery, UseQueryResult, useMutation, useQueryClient, UseMutationResult } from '@tanstack/react-query';
import { AxiosError } from 'axios';
import useAuthStore from '@/stores/useAuthStore';
import { reissueTokenApi, fetchProfileApi } from '@/api/authApi';
import { useState, useEffect } from 'react';
interface TokenResponse {
status: number;
code: number;
message: string;
data: {
accessToken: string;
};
errors: Array<{
field: string;
code: string;
message: string;
objectName: string;
}>;
isSuccess: boolean;
}
interface ProfileResponse {
status: number;
code: number;
message: string;
data: {
id: number;
nickname: string;
profileImage: string;
};
errors: Array<{
field: string;
code: string;
message: string;
objectName: string;
}>;
isSuccess: boolean;
}
interface ErrorResponse {
message: string;
}
export const useReissueToken = (): UseMutationResult<TokenResponse, AxiosError<ErrorResponse>> => {
const queryClient = useQueryClient();
const { setLoggedIn } = useAuthStore();
return useMutation({
mutationFn: reissueTokenApi,
onSuccess: (data) => {
setLoggedIn(true, data.data.accessToken);
queryClient.invalidateQueries({ queryKey: ['profile'] });
},
onError: (error) => {
console.error('토큰 재발급 실패:', error?.response?.data?.message || '알 수 없는 오류');
},
});
};
export const useProfile = (): UseQueryResult<ProfileResponse, AxiosError<ErrorResponse>> => {
const { accessToken, setProfile } = useAuthStore();
return useQuery<ProfileResponse, AxiosError<ErrorResponse>>({
queryKey: ['profile'],
queryFn: fetchProfileApi,
enabled: !!accessToken,
select: (data) => {
if (data.isSuccess) {
setProfile({
id: data.data.id,
nickname: data.data.nickname,
profileImage: data.data.profileImage,
});
}
return data;
},
});
};
export const useFetchProfile = () => {
const { profile, setProfile } = useAuthStore();
const [isFetched, setIsFetched] = useState(false);
useEffect(() => {
if (!profile.id && !isFetched) {
fetchProfileApi()
.then((data) => {
if (data?.isSuccess && data.data) {
setProfile({
id: data.data.id,
nickname: data.data.nickname,
profileImage: data.data.profileImage,
});
setIsFetched(true);
}
})
.catch((error) => {
alert('프로필을 가져오는 중 오류가 발생했습니다. 다시 시도해주세요.');
console.error('프로필 가져오기 실패:', error);
});
}
}, [profile.id, setProfile, isFetched]);
};

View File

@ -0,0 +1,150 @@
import { useQuery, UseQueryResult, useMutation, UseMutationResult, useQueryClient } from '@tanstack/react-query';
import { AxiosError } from 'axios';
import {
getProjectApi,
updateProjectApi,
deleteProjectApi,
getAllProjectsApi,
createProjectApi,
addProjectMemberApi,
removeProjectMemberApi,
} from '@/api/projectApi';
interface Project {
id: number;
title: string;
workspaceId: number;
projectType: string;
createdAt: string;
updatedAt: string;
}
interface ProjectsResponse {
status: number;
code: number;
message: string;
data: {
workspaceResponses: Project[];
};
errors: Array<{
field: string;
code: string;
message: string;
objectName: string;
}>;
isSuccess: boolean;
}
interface ErrorResponse {
message: string;
}
export const useGetProject = (
projectId: number,
memberId: number
): UseQueryResult<ProjectsResponse, AxiosError<ErrorResponse>> => {
return useQuery<ProjectsResponse, AxiosError<ErrorResponse>>({
queryKey: ['project', projectId],
queryFn: () => getProjectApi(projectId, memberId),
});
};
export const useUpdateProject = (): UseMutationResult<
ProjectsResponse,
AxiosError<ErrorResponse>,
{ projectId: number; memberId: number; data: { title: string; projectType: string } }
> => {
const queryClient = useQueryClient();
return useMutation({
mutationFn: ({ projectId, memberId, data }) => updateProjectApi(projectId, memberId, data),
onSuccess: (data) => {
const project = data.data?.workspaceResponses?.[0];
if (project) {
queryClient.invalidateQueries({ queryKey: ['project', project.id] });
} else {
console.error('프로젝트 데이터가 없습니다.');
}
},
});
};
export const useDeleteProject = (): UseMutationResult<
ProjectsResponse,
AxiosError<ErrorResponse>,
{ projectId: number; memberId: number }
> => {
const queryClient = useQueryClient();
return useMutation({
mutationFn: ({ projectId, memberId }) => deleteProjectApi(projectId, memberId),
onSuccess: (data) => {
const project = data.data?.workspaceResponses?.[0];
if (project) {
queryClient.invalidateQueries({ queryKey: ['project', project.id] });
} else {
console.error('프로젝트 데이터가 없습니다.');
}
},
});
};
export const useGetAllProjects = (
workspaceId: number,
memberId: number,
lastProjectId?: number,
limit: number = 10
): UseQueryResult<ProjectsResponse, AxiosError<ErrorResponse>> => {
return useQuery<ProjectsResponse, AxiosError<ErrorResponse>>({
queryKey: ['projects', workspaceId],
queryFn: () => getAllProjectsApi(workspaceId, memberId, lastProjectId, limit),
});
};
export const useCreateProject = (): UseMutationResult<
ProjectsResponse,
AxiosError<ErrorResponse>,
{ workspaceId: number; memberId: number; data: { title: string; projectType: string } }
> => {
return useMutation({
mutationFn: ({ workspaceId, memberId, data }) => createProjectApi(workspaceId, memberId, data),
});
};
export const useAddProjectMember = (): UseMutationResult<
ProjectsResponse,
AxiosError<ErrorResponse>,
{ projectId: number; memberId: number; newMemberId: number; privilegeType: string }
> => {
const queryClient = useQueryClient();
return useMutation({
mutationFn: ({ projectId, memberId, newMemberId, privilegeType }) =>
addProjectMemberApi(projectId, memberId, newMemberId, privilegeType),
onSuccess: (data) => {
const project = data.data?.workspaceResponses?.[0];
if (project) {
queryClient.invalidateQueries({ queryKey: ['project', project.id] });
} else {
console.error('프로젝트 데이터가 없습니다.');
}
},
});
};
export const useRemoveProjectMember = (): UseMutationResult<
ProjectsResponse,
AxiosError<ErrorResponse>,
{ projectId: number; memberId: number; targetMemberId: number }
> => {
const queryClient = useQueryClient();
return useMutation({
mutationFn: ({ projectId, memberId, targetMemberId }) =>
removeProjectMemberApi(projectId, memberId, targetMemberId),
onSuccess: (data) => {
const project = data.data?.workspaceResponses?.[0];
if (project) {
queryClient.invalidateQueries({ queryKey: ['project', project.id] });
} else {
console.error('프로젝트 데이터가 없습니다.');
}
},
});
};

View File

@ -0,0 +1,152 @@
import { useQuery, UseQueryResult, useMutation, UseMutationResult, useQueryClient } from '@tanstack/react-query';
import { AxiosError } from 'axios';
import {
getWorkspaceApi,
updateWorkspaceApi,
deleteWorkspaceApi,
getAllWorkspacesApi,
createWorkspaceApi,
addWorkspaceMemberApi,
removeWorkspaceMemberApi,
} from '@/api/workspaceApi';
interface WorkspaceResponse {
status: number;
code: number;
message: string;
data: {
id: number;
memberId: string;
title: string;
content: string;
createdAt: string;
updatedAt: string;
};
errors: Array<{
field: string;
code: string;
message: string;
objectName: string;
}>;
isSuccess: boolean;
}
interface Workspace {
id: number;
memberId: string;
title: string;
content: string;
createdAt: string;
updatedAt: string;
}
interface GetAllWorkspacesResponse {
status: number;
code: number;
message: string;
data: {
workspaceResponses: Workspace[];
};
errors: Array<{
field: string;
code: string;
message: string;
objectName: string;
}>;
isSuccess: boolean;
}
interface ErrorResponse {
message: string;
}
export const useGetWorkspace = (
workspaceId: number,
memberId: number
): UseQueryResult<WorkspaceResponse, AxiosError<ErrorResponse>> => {
return useQuery<WorkspaceResponse, AxiosError<ErrorResponse>>({
queryKey: ['workspace', workspaceId],
queryFn: () => getWorkspaceApi(workspaceId, memberId),
});
};
export const useUpdateWorkspace = (): UseMutationResult<
WorkspaceResponse,
AxiosError<ErrorResponse>,
{ workspaceId: number; memberId: number; data: { title: string; content: string } }
> => {
const queryClient = useQueryClient();
return useMutation({
mutationFn: ({ workspaceId, memberId, data }) => updateWorkspaceApi(workspaceId, memberId, data),
onSuccess: () => {
queryClient.invalidateQueries({ queryKey: ['workspace'] });
},
});
};
export const useDeleteWorkspace = (): UseMutationResult<
WorkspaceResponse,
AxiosError<ErrorResponse>,
{ workspaceId: number; memberId: number }
> => {
const queryClient = useQueryClient();
return useMutation({
mutationFn: ({ workspaceId, memberId }) => deleteWorkspaceApi(workspaceId, memberId),
onSuccess: () => {
queryClient.invalidateQueries({ queryKey: ['workspace'] });
},
});
};
export const useGetAllWorkspaces = (
memberId: number,
lastWorkspaceId?: number,
limit?: number
): UseQueryResult<GetAllWorkspacesResponse, AxiosError<ErrorResponse>> => {
return useQuery<GetAllWorkspacesResponse, AxiosError<ErrorResponse>>({
queryKey: ['workspaces'],
queryFn: () => getAllWorkspacesApi(memberId, lastWorkspaceId, limit),
});
};
export const useCreateWorkspace = (): UseMutationResult<
WorkspaceResponse,
AxiosError<ErrorResponse>,
{ memberId: number; data: { title: string; content: string } }
> => {
const queryClient = useQueryClient();
return useMutation({
mutationFn: ({ memberId, data }) => createWorkspaceApi(memberId, data),
onSuccess: () => {
queryClient.invalidateQueries({ queryKey: ['workspace'] });
},
});
};
export const useAddWorkspaceMember = (): UseMutationResult<
WorkspaceResponse,
AxiosError<ErrorResponse>,
{ workspaceId: number; memberId: number; newMemberId: number }
> => {
const queryClient = useQueryClient();
return useMutation({
mutationFn: ({ workspaceId, memberId, newMemberId }) => addWorkspaceMemberApi(workspaceId, memberId, newMemberId),
onSuccess: () => {
queryClient.invalidateQueries({ queryKey: ['workspace'] });
},
});
};
export const useRemoveWorkspaceMember = (): UseMutationResult<
WorkspaceResponse,
AxiosError<ErrorResponse>,
{ workspaceId: number; memberId: number; targetMemberId: number }
> => {
const queryClient = useQueryClient();
return useMutation({
mutationFn: ({ workspaceId, memberId, targetMemberId }) =>
removeWorkspaceMemberApi(workspaceId, memberId, targetMemberId),
onSuccess: () => {
queryClient.invalidateQueries({ queryKey: ['workspace'] });
},
});
};

View File

@ -1,3 +1,30 @@
// import React from 'react';
// import ReactDOM from 'react-dom/client';
// import App from './App.tsx';
// import './index.css';
// async function enableMocking() {
// if (!import.meta.env.DEV) {
// return;
// }
// try {
// const { worker } = await import('./mocks/browser.ts');
// await worker.start();
// console.log('[MSW] Mocking enabled. Service Worker is running.');
// } catch (error) {
// console.error('[MSW] Failed to start the Service Worker:', error);
// }
// }
// enableMocking().then(() => {
// ReactDOM.createRoot(document.getElementById('root')!).render(
// <React.StrictMode>
// <App />
// </React.StrictMode>
// );
// });
import React from 'react';
import ReactDOM from 'react-dom/client';
import App from './App.tsx';

View File

@ -0,0 +1,4 @@
import { setupWorker } from 'msw/browser';
import { handlers } from './handlers';
export const worker = setupWorker(...handlers);

View File

@ -0,0 +1,267 @@
import { http, HttpResponse } from 'msw';
export const handlers = [
http.post('/api/auth/reissue', () => {
return HttpResponse.json({
status: 0,
code: 0,
message: '토큰 재발급 성공',
data: { accessToken: 'mockAccessToken' },
errors: [],
isSuccess: true,
});
}),
http.get('/api/auth/profile', () => {
return HttpResponse.json({
status: 0,
code: 0,
message: '프로필 조회 성공',
data: {
id: 1,
nickname: 'mockUser',
profileImage: 'mockImage.jpg',
},
errors: [],
isSuccess: true,
});
}),
http.get('/api/projects/:projectId', ({ params }) => {
const { projectId } = params;
return HttpResponse.json({
status: 0,
code: 0,
message: '프로젝트 조회 성공',
data: {
id: Number(projectId),
title: `Project ${projectId}`,
workspaceId: 1,
projectType: 'classification',
createdAt: '2024-09-10T04:01:37.033Z',
updatedAt: '2024-09-10T04:01:37.033Z',
},
errors: [],
isSuccess: true,
});
}),
http.put('/api/projects/:projectId', async ({ params, request }) => {
const { projectId } = params;
const updateData = (await request.json()) as Record<string, unknown>;
return HttpResponse.json({
status: 0,
code: 0,
message: '프로젝트 수정 성공',
data: {
id: Number(projectId),
...updateData,
workspaceId: 1,
createdAt: '2024-09-10T04:01:37.033Z',
updatedAt: new Date().toISOString(),
},
errors: [],
isSuccess: true,
});
}),
http.delete('/api/projects/:projectId', ({ params }) => {
const { projectId } = params;
return HttpResponse.json({
status: 0,
code: 0,
message: `프로젝트 ${projectId} 삭제 성공`,
data: null,
errors: [],
isSuccess: true,
});
}),
http.post('/api/projects/:projectId/members', async ({ params, request }) => {
const { projectId } = params;
const memberData = (await request.json()) as Record<string, unknown>;
return HttpResponse.json({
status: 0,
code: 0,
message: `프로젝트 ${projectId} 멤버 추가 성공`,
data: { ...memberData },
errors: [],
isSuccess: true,
});
}),
http.delete('/api/projects/:projectId/members', async ({ params, request }) => {
const { projectId } = params;
const memberData = (await request.json()) as Record<string, unknown>;
return HttpResponse.json({
status: 0,
code: 0,
message: `프로젝트 ${projectId} 멤버 제거 성공`,
data: { ...memberData },
errors: [],
isSuccess: true,
});
}),
http.get('/api/workspaces/:workspaceId/projects', async ({ params, request }) => {
const { workspaceId } = params;
const url = new URL(request.url);
const memberId = url.searchParams.get('memberId');
if (!memberId) {
return HttpResponse.json({
status: 1,
code: 400,
message: 'memberId가 필요합니다.',
data: null,
errors: [{ field: 'memberId', code: 'missing', message: 'memberId가 필요합니다.', objectName: 'request' }],
isSuccess: false,
});
}
const dummyProjects = [
{
id: 1,
title: `Project 1`,
workspaceId: Number(workspaceId),
projectType: 'classification',
createdAt: '2024-09-10T04:01:37.049Z',
updatedAt: '2024-09-10T04:01:37.049Z',
},
{
id: 2,
title: `Project 2`,
workspaceId: Number(workspaceId),
projectType: 'detection',
createdAt: '2024-09-10T04:02:37.049Z',
updatedAt: '2024-09-10T04:02:37.049Z',
},
];
return HttpResponse.json({
status: 0,
code: 0,
message: '프로젝트 목록 조회 성공',
data: {
workspaceResponses: dummyProjects, // Adjusted structure to match the expected response type
},
errors: [],
isSuccess: true,
});
}),
http.post('/api/workspaces/:workspaceId/projects', async ({ params, request }) => {
const { workspaceId } = params;
const newProject = (await request.json()) as Record<string, unknown>;
return HttpResponse.json({
status: 0,
code: 0,
message: '프로젝트 생성 성공',
data: {
...newProject,
workspaceId: Number(workspaceId),
id: Math.floor(Math.random() * 1000),
createdAt: new Date().toISOString(),
updatedAt: new Date().toISOString(),
},
errors: [],
isSuccess: true,
});
}),
http.get('/api/workspaces/:workspaceId', ({ params }) => {
const { workspaceId } = params;
return HttpResponse.json({
status: 0,
code: 0,
message: '워크스페이스 조회 성공',
data: {
id: Number(workspaceId),
memberId: 'member123',
title: 'workspace1',
content: '갤럭시 s24 불량 검증',
createdAt: '2024-09-10T03:46:17.421Z',
updatedAt: '2024-09-10T03:46:17.421Z',
},
errors: [],
isSuccess: true,
});
}),
http.put('/api/workspaces/:workspaceId', async ({ params, request }) => {
const { workspaceId } = params;
const updateData = (await request.json()) as Record<string, unknown>;
return HttpResponse.json({
status: 0,
code: 0,
message: '워크스페이스 수정 성공',
data: {
id: Number(workspaceId),
...updateData,
createdAt: '2024-09-10T03:46:17.421Z',
updatedAt: new Date().toISOString(),
},
errors: [],
isSuccess: true,
});
}),
http.delete('/api/workspaces/:workspaceId', ({ params }) => {
const { workspaceId } = params;
return HttpResponse.json({
status: 0,
code: 0,
message: `워크스페이스 ${workspaceId} 삭제 성공`,
data: null,
errors: [],
isSuccess: true,
});
}),
http.get('/api/workspaces', () => {
return HttpResponse.json({
status: 0,
code: 0,
message: '전체 워크스페이스 조회 성공',
data: {
workspaceResponses: [
{
id: 1,
memberId: 'member123',
title: 'workspace1',
content: '갤럭시 s24 불량 검증',
createdAt: '2024-09-10T03:46:17.428Z',
updatedAt: '2024-09-10T03:46:17.428Z',
},
{
id: 2,
memberId: 'member123',
title: 'workspace2',
content: '갤럭시 s24 불량 검증',
createdAt: '2024-09-10T03:46:17.428Z',
updatedAt: '2024-09-10T03:46:17.428Z',
},
],
},
errors: [],
isSuccess: true,
});
}),
http.post('/api/workspaces', async ({ request }) => {
const newWorkspace = (await request.json()) as Record<string, unknown>;
return HttpResponse.json({
status: 0,
code: 0,
message: '워크스페이스 생성 성공',
data: {
...newWorkspace,
id: Math.floor(Math.random() * 1000),
createdAt: new Date().toISOString(),
updatedAt: new Date().toISOString(),
},
errors: [],
isSuccess: true,
});
}),
];

View File

@ -6,6 +6,7 @@ import WorkspaceLayout from '@/components/WorkspaceLayout';
import AdminLayout from '@/components/AdminLayout';
import ReviewList from '@/components/ReviewList';
import AdminMemberManage from '@/components/AdminMemberManage';
import OAuthCallback from '@/components/OAuthCallback';
import { createBrowserRouter } from 'react-router-dom';
import { Navigate } from 'react-router-dom';
@ -14,6 +15,7 @@ export const webPath = {
browse: () => '/browse',
workspace: () => '/workspace',
admin: (id: string) => `/admin/${id}`,
oauthCallback: () => '/redirect/oauth2',
};
const router = createBrowserRouter([
@ -69,6 +71,10 @@ const router = createBrowserRouter([
},
],
},
{
path: webPath.oauthCallback(),
element: <OAuthCallback />,
},
]);
export default router;

View File

@ -0,0 +1,37 @@
import { create } from 'zustand';
import { persist } from 'zustand/middleware';
interface ProfileData {
id: number | null;
nickname: string;
profileImage: string;
}
interface AuthState {
isLoggedIn: boolean;
accessToken: string;
profile: ProfileData;
setLoggedIn: (status: boolean, token: string) => void;
setProfile: (profile: ProfileData) => void;
clearAuth: () => void;
}
const useAuthStore = create<AuthState>()(
persist(
(set) => ({
isLoggedIn: false,
accessToken: '',
profile: { id: null, nickname: '', profileImage: '' },
setLoggedIn: (status, token) => set({ isLoggedIn: status, accessToken: token }),
setProfile: (profile) => set({ profile }),
clearAuth: () =>
set({ isLoggedIn: false, accessToken: '', profile: { id: null, nickname: '', profileImage: '' } }),
}),
{
name: 'auth-storage',
getStorage: () => localStorage,
}
)
);
export default useAuthStore;