22 lines
19 KiB
JavaScript
22 lines
19 KiB
JavaScript
"use strict";
|
|
/*
|
|
* ATTENTION: An "eval-source-map" devtool has been used.
|
|
* This devtool is neither made for production nor for readable output files.
|
|
* It uses "eval()" calls to create a separate source file with attached SourceMaps in the browser devtools.
|
|
* If you are trying to read the output file, select a different devtool (https://webpack.js.org/configuration/devtool/)
|
|
* or disable the default devtool with "devtool: false".
|
|
* If you are looking for production-ready output files, see mode: "production" (https://webpack.js.org/configuration/mode/).
|
|
*/
|
|
self["webpackHotUpdate_N_E"]("app/layout",{
|
|
|
|
/***/ "(app-pages-browser)/./lib/services/auth.ts":
|
|
/*!******************************!*\
|
|
!*** ./lib/services/auth.ts ***!
|
|
\******************************/
|
|
/***/ ((module, __webpack_exports__, __webpack_require__) => {
|
|
|
|
eval(__webpack_require__.ts("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ authService: () => (/* binding */ authService)\n/* harmony export */ });\n// Auth service for handling authentication API calls\nconst API_BASE_URL = 'http://localhost:5000';\nclass AuthService {\n async makeRequest(endpoint) {\n let options = arguments.length > 1 && arguments[1] !== void 0 ? arguments[1] : {};\n const url = \"\".concat(API_BASE_URL, \"/api/auth\").concat(endpoint);\n console.log('Making request to:', url);\n console.log('Request body:', options.body);\n const response = await fetch(url, {\n headers: {\n 'Content-Type': 'application/json',\n ...options.headers\n },\n ...options\n });\n if (!response.ok) {\n const errorData = await response.json().catch(()=>({\n error: 'Network error'\n }));\n console.log('API Error Response:', errorData);\n // 處理驗證錯誤 (400 Bad Request)\n if (response.status === 400 && errorData.errors) {\n const validationErrors = Object.entries(errorData.errors).map((param)=>{\n let [field, messages] = param;\n return \"\".concat(field, \": \").concat(Array.isArray(messages) ? messages.join(', ') : messages);\n }).join('\\n');\n throw new Error(validationErrors);\n }\n throw new Error(errorData.error || errorData.title || \"HTTP \".concat(response.status));\n }\n return response.json();\n }\n async login(data) {\n try {\n var _response_data;\n const response = await this.makeRequest('/login', {\n method: 'POST',\n body: JSON.stringify(data)\n });\n if (response.success && ((_response_data = response.data) === null || _response_data === void 0 ? void 0 : _response_data.token)) {\n // Store token in localStorage\n localStorage.setItem('auth_token', response.data.token);\n localStorage.setItem('user_data', JSON.stringify(response.data.user));\n }\n return response;\n } catch (error) {\n return {\n success: false,\n error: error instanceof Error ? error.message : 'Login failed'\n };\n }\n }\n async register(data) {\n try {\n var _response_data;\n console.log('AuthService.register called with:', {\n username: data.username,\n email: data.email,\n password: data.password ? '[hidden]' : 'empty'\n });\n const response = await this.makeRequest('/register', {\n method: 'POST',\n body: JSON.stringify(data)\n });\n if (response.success && ((_response_data = response.data) === null || _response_data === void 0 ? void 0 : _response_data.token)) {\n // Store token in localStorage\n localStorage.setItem('auth_token', response.data.token);\n localStorage.setItem('user_data', JSON.stringify(response.data.user));\n }\n return response;\n } catch (error) {\n return {\n success: false,\n error: error instanceof Error ? error.message : 'Registration failed'\n };\n }\n }\n async logout() {\n localStorage.removeItem('auth_token');\n localStorage.removeItem('user_data');\n }\n getToken() {\n if (false) {}\n return localStorage.getItem('auth_token');\n }\n getUser() {\n if (false) {}\n const userData = localStorage.getItem('user_data');\n return userData ? JSON.parse(userData) : null;\n }\n isAuthenticated() {\n return this.getToken() !== null;\n }\n async checkAuthStatus() {\n const token = this.getToken();\n if (!token) return false;\n try {\n await this.makeRequest('/status', {\n headers: {\n 'Authorization': \"Bearer \".concat(token)\n }\n });\n return true;\n } catch (e) {\n // Token is invalid, clear it\n this.logout();\n return false;\n }\n }\n}\nconst authService = new AuthService();\n\n\n;\n // Wrapped in an IIFE to avoid polluting the global scope\n ;\n (function () {\n var _a, _b;\n // Legacy CSS implementations will `eval` browser code in a Node.js context\n // to extract CSS. For backwards compatibility, we need to check we're in a\n // browser context before continuing.\n if (typeof self !== 'undefined' &&\n // AMP / No-JS mode does not inject these helpers:\n '$RefreshHelpers$' in self) {\n // @ts-ignore __webpack_module__ is global\n var currentExports = module.exports;\n // @ts-ignore __webpack_module__ is global\n var prevSignature = (_b = (_a = module.hot.data) === null || _a === void 0 ? void 0 : _a.prevSignature) !== null && _b !== void 0 ? _b : null;\n // This cannot happen in MainTemplate because the exports mismatch between\n // templating and execution.\n self.$RefreshHelpers$.registerExportsForReactRefresh(currentExports, module.id);\n // A module can be accepted automatically based on its exports, e.g. when\n // it is a Refresh Boundary.\n if (self.$RefreshHelpers$.isReactRefreshBoundary(currentExports)) {\n // Save the previous exports signature on update so we can compare the boundary\n // signatures. We avoid saving exports themselves since it causes memory leaks (https://github.com/vercel/next.js/pull/53797)\n module.hot.dispose(function (data) {\n data.prevSignature =\n self.$RefreshHelpers$.getRefreshBoundarySignature(currentExports);\n });\n // Unconditionally accept an update to this module, we'll check if it's\n // still a Refresh Boundary later.\n // @ts-ignore importMeta is replaced in the loader\n module.hot.accept();\n // This field is set when the previous version of this module was a\n // Refresh Boundary, letting us know we need to check for invalidation or\n // enqueue an update.\n if (prevSignature !== null) {\n // A boundary can become ineligible if its exports are incompatible\n // with the previous exports.\n //\n // For example, if you add/remove/change exports, we'll want to\n // re-execute the importing modules, and force those components to\n // re-render. Similarly, if you convert a class component to a\n // function, we want to invalidate the boundary.\n if (self.$RefreshHelpers$.shouldInvalidateReactRefreshBoundary(prevSignature, self.$RefreshHelpers$.getRefreshBoundarySignature(currentExports))) {\n module.hot.invalidate();\n }\n else {\n self.$RefreshHelpers$.scheduleUpdate();\n }\n }\n }\n else {\n // Since we just executed the code for the module, it's possible that the\n // new exports made it ineligible for being a boundary.\n // We only care about the case when we were _previously_ a boundary,\n // because we already accepted this update (accidental side effect).\n var isNoLongerABoundary = prevSignature !== null;\n if (isNoLongerABoundary) {\n module.hot.invalidate();\n }\n }\n }\n })();\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"(app-pages-browser)/./lib/services/auth.ts","mappings":"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","sources":["/Users/jettcheng1018/code/dramaling-vocab-learning/frontend/lib/services/auth.ts"],"sourcesContent":["// Auth service for handling authentication API calls\r\n\r\nexport interface LoginRequest {\r\n  email: string;\r\n  password: string;\r\n}\r\n\r\nexport interface RegisterRequest {\r\n  username: string;\r\n  email: string;\r\n  password: string;\r\n}\r\n\r\nexport interface User {\r\n  id: string;\r\n  username: string;\r\n  email: string;\r\n  displayName: string | null;\r\n  avatarUrl: string | null;\r\n  subscriptionType: string;\r\n}\r\n\r\nexport interface AuthResponse {\r\n  success: boolean;\r\n  data?: {\r\n    token: string;\r\n    user: User;\r\n  };\r\n  error?: string;\r\n}\r\n\r\nconst API_BASE_URL = 'http://localhost:5000';\r\n\r\nclass AuthService {\r\n  private async makeRequest<T>(\r\n    endpoint: string,\r\n    options: RequestInit = {}\r\n  ): Promise<T> {\r\n    const url = `${API_BASE_URL}/api/auth${endpoint}`;\r\n\r\n    console.log('Making request to:', url)\r\n    console.log('Request body:', options.body)\r\n\r\n    const response = await fetch(url, {\r\n      headers: {\r\n        'Content-Type': 'application/json',\r\n        ...options.headers,\r\n      },\r\n      ...options,\r\n    });\r\n\r\n    if (!response.ok) {\r\n      const errorData = await response.json().catch(() => ({ error: 'Network error' }));\r\n      console.log('API Error Response:', errorData)\r\n\r\n      // 處理驗證錯誤 (400 Bad Request)\r\n      if (response.status === 400 && errorData.errors) {\r\n        const validationErrors = Object.entries(errorData.errors)\r\n          .map(([field, messages]: [string, any]) => `${field}: ${Array.isArray(messages) ? messages.join(', ') : messages}`)\r\n          .join('\\n');\r\n        throw new Error(validationErrors);\r\n      }\r\n\r\n      throw new Error(errorData.error || errorData.title || `HTTP ${response.status}`);\r\n    }\r\n\r\n    return response.json();\r\n  }\r\n\r\n  async login(data: LoginRequest): Promise<AuthResponse> {\r\n    try {\r\n      const response = await this.makeRequest<AuthResponse>('/login', {\r\n        method: 'POST',\r\n        body: JSON.stringify(data),\r\n      });\r\n\r\n      if (response.success && response.data?.token) {\r\n        // Store token in localStorage\r\n        localStorage.setItem('auth_token', response.data.token);\r\n        localStorage.setItem('user_data', JSON.stringify(response.data.user));\r\n      }\r\n\r\n      return response;\r\n    } catch (error) {\r\n      return {\r\n        success: false,\r\n        error: error instanceof Error ? error.message : 'Login failed',\r\n      };\r\n    }\r\n  }\r\n\r\n  async register(data: RegisterRequest): Promise<AuthResponse> {\r\n    try {\r\n      console.log('AuthService.register called with:', {\r\n        username: data.username,\r\n        email: data.email,\r\n        password: data.password ? '[hidden]' : 'empty'\r\n      })\r\n\r\n      const response = await this.makeRequest<AuthResponse>('/register', {\r\n        method: 'POST',\r\n        body: JSON.stringify(data),\r\n      });\r\n\r\n      if (response.success && response.data?.token) {\r\n        // Store token in localStorage\r\n        localStorage.setItem('auth_token', response.data.token);\r\n        localStorage.setItem('user_data', JSON.stringify(response.data.user));\r\n      }\r\n\r\n      return response;\r\n    } catch (error) {\r\n      return {\r\n        success: false,\r\n        error: error instanceof Error ? error.message : 'Registration failed',\r\n      };\r\n    }\r\n  }\r\n\r\n  async logout(): Promise<void> {\r\n    localStorage.removeItem('auth_token');\r\n    localStorage.removeItem('user_data');\r\n  }\r\n\r\n  getToken(): string | null {\r\n    if (typeof window === 'undefined') return null;\r\n    return localStorage.getItem('auth_token');\r\n  }\r\n\r\n  getUser(): User | null {\r\n    if (typeof window === 'undefined') return null;\r\n    const userData = localStorage.getItem('user_data');\r\n    return userData ? JSON.parse(userData) : null;\r\n  }\r\n\r\n  isAuthenticated(): boolean {\r\n    return this.getToken() !== null;\r\n  }\r\n\r\n  async checkAuthStatus(): Promise<boolean> {\r\n    const token = this.getToken();\r\n    if (!token) return false;\r\n\r\n    try {\r\n      await this.makeRequest('/status', {\r\n        headers: {\r\n          'Authorization': `Bearer ${token}`,\r\n        },\r\n      });\r\n      return true;\r\n    } catch {\r\n      // Token is invalid, clear it\r\n      this.logout();\r\n      return false;\r\n    }\r\n  }\r\n}\r\n\r\nexport const authService = new AuthService();"],"names":["API_BASE_URL","AuthService","makeRequest","endpoint","options","url","console","log","body","response","fetch","headers","ok","errorData","json","catch","error","status","errors","validationErrors","Object","entries","map","field","messages","Array","isArray","join","Error","title","login","data","method","JSON","stringify","success","token","localStorage","setItem","user","message","register","username","email","password","logout","removeItem","getToken","getItem","getUser","userData","parse","isAuthenticated","checkAuthStatus","authService"],"ignoreList":[],"sourceRoot":""}\n//# sourceURL=webpack-internal:///(app-pages-browser)/./lib/services/auth.ts\n"));
|
|
|
|
/***/ })
|
|
|
|
}); |