diff --git a/package-lock.json b/package-lock.json index 5ac6f652665ccf93a11e00cd912be8ef55f7d8ae..8749b6a2570d295149b044ecb085db94100a5753 100644 --- a/package-lock.json +++ b/package-lock.json @@ -14,6 +14,7 @@ "install": "^0.13.0", "js-cookie": "^3.0.5", "oh-vue-icons": "^1.0.0-rc3", + "openapi-typescript-codegen": "^0.29.0", "pinia": "^2.1.7", "pinia-plugin-persist": "^1.0.0", "pinia-plugin-persistedstate": "^3.2.1", @@ -87,6 +88,22 @@ "url": "https://github.com/sponsors/antfu" } }, + "node_modules/@apidevtools/json-schema-ref-parser": { + "version": "11.6.1", + "resolved": "https://registry.npmjs.org/@apidevtools/json-schema-ref-parser/-/json-schema-ref-parser-11.6.1.tgz", + "integrity": "sha512-DxjgKBCoyReu4p5HMvpmgSOfRhhBcuf5V5soDDRgOTZMwsA4KSFzol1abFZgiCTE11L2kKGca5Md9GwDdXVBwQ==", + "dependencies": { + "@jsdevtools/ono": "^7.1.3", + "@types/json-schema": "^7.0.15", + "js-yaml": "^4.1.0" + }, + "engines": { + "node": ">= 16" + }, + "funding": { + "url": "https://github.com/sponsors/philsturgeon" + } + }, "node_modules/@babel/code-frame": { "version": "7.24.2", "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.24.2.tgz", @@ -1418,6 +1435,11 @@ "@jridgewell/sourcemap-codec": "^1.4.14" } }, + "node_modules/@jsdevtools/ono": { + "version": "7.1.3", + "resolved": "https://registry.npmjs.org/@jsdevtools/ono/-/ono-7.1.3.tgz", + "integrity": "sha512-4JQNk+3mVzK3xh2rqd6RB4J46qUR19azEHBneZyTZM+c456qOrbbM/5xcR8huNCCcbVt7+UmizG6GuUvPvKUYg==" + }, "node_modules/@nodelib/fs.scandir": { "version": "2.1.5", "resolved": "https://registry.npmjs.org/@nodelib/fs.scandir/-/fs.scandir-2.1.5.tgz", @@ -1891,8 +1913,7 @@ "node_modules/@types/json-schema": { "version": "7.0.15", "resolved": "https://registry.npmjs.org/@types/json-schema/-/json-schema-7.0.15.tgz", - "integrity": "sha512-5+fP8P8MFNC+AyZCDxrB2pkZFPGzqQWUzpSeuuVLvm8VMcorNYavBqoFcxK8bQz4Qsbn4oUEEem4wDLfcysGHA==", - "dev": true + "integrity": "sha512-5+fP8P8MFNC+AyZCDxrB2pkZFPGzqQWUzpSeuuVLvm8VMcorNYavBqoFcxK8bQz4Qsbn4oUEEem4wDLfcysGHA==" }, "node_modules/@types/node": { "version": "20.12.7", @@ -2928,8 +2949,7 @@ "node_modules/argparse": { "version": "2.0.1", "resolved": "https://registry.npmjs.org/argparse/-/argparse-2.0.1.tgz", - "integrity": "sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==", - "dev": true + "integrity": "sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==" }, "node_modules/aria-query": { "version": "5.3.0", @@ -3315,7 +3335,6 @@ "version": "6.3.0", "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-6.3.0.tgz", "integrity": "sha512-Gmy6FhYlCY7uOElZUSbxo2UCDH8owEk996gkbrpsgGtrJLM3J7jGxl9Ic7Qwwj4ivOE5AWZWRMecDdF7hqGjFA==", - "dev": true, "engines": { "node": ">=10" }, @@ -5355,8 +5374,7 @@ "node_modules/graceful-fs": { "version": "4.2.11", "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.11.tgz", - "integrity": "sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ==", - "dev": true + "integrity": "sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ==" }, "node_modules/graphemer": { "version": "1.4.0", @@ -5364,6 +5382,26 @@ "integrity": "sha512-EtKwoO6kxCL9WO5xipiHTZlSzBm7WLT627TqC/uVRd0HKmq8NXyebnNYxDoBi7wt8eTWrUrKXCOVaFq9x1kgag==", "dev": true }, + "node_modules/handlebars": { + "version": "4.7.8", + "resolved": "https://registry.npmjs.org/handlebars/-/handlebars-4.7.8.tgz", + "integrity": "sha512-vafaFqs8MZkRrSX7sFVUdo3ap/eNiLnb4IakshzvP56X5Nr1iGKAIqdX6tMlm6HcNRIkr6AxO5jFEoJzzpT8aQ==", + "dependencies": { + "minimist": "^1.2.5", + "neo-async": "^2.6.2", + "source-map": "^0.6.1", + "wordwrap": "^1.0.0" + }, + "bin": { + "handlebars": "bin/handlebars" + }, + "engines": { + "node": ">=0.4.7" + }, + "optionalDependencies": { + "uglify-js": "^3.1.4" + } + }, "node_modules/has-bigints": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/has-bigints/-/has-bigints-1.0.2.tgz", @@ -6252,7 +6290,6 @@ "version": "4.1.0", "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-4.1.0.tgz", "integrity": "sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==", - "dev": true, "dependencies": { "argparse": "^2.0.1" }, @@ -6388,7 +6425,6 @@ "version": "6.1.0", "resolved": "https://registry.npmjs.org/jsonfile/-/jsonfile-6.1.0.tgz", "integrity": "sha512-5dgndWOriYSm5cnYaJNhalLNDKOqFwyDB/rr1E9ZsGciGvKPs8R2xYGCacuf3z6K1YKDz182fd+fY3cn3pMqXQ==", - "dev": true, "dependencies": { "universalify": "^2.0.0" }, @@ -6814,7 +6850,6 @@ "version": "1.2.8", "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.8.tgz", "integrity": "sha512-2yyAR8qBkN3YuheJanUpWC5U3bb5osDywNB8RzDVlDwDHbocAJveqqj1u8+SVD7jkWT4yvsHCpWqqWqAxb0zCA==", - "dev": true, "funding": { "url": "https://github.com/sponsors/ljharb" } @@ -6902,6 +6937,11 @@ "integrity": "sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw==", "dev": true }, + "node_modules/neo-async": { + "version": "2.6.2", + "resolved": "https://registry.npmjs.org/neo-async/-/neo-async-2.6.2.tgz", + "integrity": "sha512-Yd3UES5mWCSqR+qNT93S3UoYUkqAZ9lLg8a7g9rimsWmYGK8cVToA4/sF3RrshdyV3sAGMXVUmpMYOw+dLpOuw==" + }, "node_modules/nice-try": { "version": "1.0.5", "resolved": "https://registry.npmjs.org/nice-try/-/nice-try-1.0.5.tgz", @@ -7359,6 +7399,42 @@ "url": "https://github.com/sponsors/sindresorhus" } }, + "node_modules/openapi-typescript-codegen": { + "version": "0.29.0", + "resolved": "https://registry.npmjs.org/openapi-typescript-codegen/-/openapi-typescript-codegen-0.29.0.tgz", + "integrity": "sha512-/wC42PkD0LGjDTEULa/XiWQbv4E9NwLjwLjsaJ/62yOsoYhwvmBR31kPttn1DzQ2OlGe5stACcF/EIkZk43M6w==", + "dependencies": { + "@apidevtools/json-schema-ref-parser": "^11.5.4", + "camelcase": "^6.3.0", + "commander": "^12.0.0", + "fs-extra": "^11.2.0", + "handlebars": "^4.7.8" + }, + "bin": { + "openapi": "bin/index.js" + } + }, + "node_modules/openapi-typescript-codegen/node_modules/commander": { + "version": "12.0.0", + "resolved": "https://registry.npmjs.org/commander/-/commander-12.0.0.tgz", + "integrity": "sha512-MwVNWlYjDTtOjX5PiD7o5pK0UrFU/OYgcJfjjK4RaHZETNtjJqrZa9Y9ds88+A+f+d5lv+561eZ+yCKoS3gbAA==", + "engines": { + "node": ">=18" + } + }, + "node_modules/openapi-typescript-codegen/node_modules/fs-extra": { + "version": "11.2.0", + "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-11.2.0.tgz", + "integrity": "sha512-PmDi3uwK5nFuXh7XDTlVnS17xJS7vW36is2+w3xcv8SVxiB4NyATf4ctkVY5bkSjX0Y4nbvZCq1/EjtEyr9ktw==", + "dependencies": { + "graceful-fs": "^4.2.0", + "jsonfile": "^6.0.1", + "universalify": "^2.0.0" + }, + "engines": { + "node": ">=14.14" + } + }, "node_modules/optionator": { "version": "0.9.3", "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.9.3.tgz", @@ -8459,6 +8535,14 @@ "node": ">=8" } }, + "node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "engines": { + "node": ">=0.10.0" + } + }, "node_modules/source-map-js": { "version": "1.2.0", "resolved": "https://registry.npmjs.org/source-map-js/-/source-map-js-1.2.0.tgz", @@ -9415,6 +9499,18 @@ "integrity": "sha512-Y7HYmWaFwPUmkoQCUIAYpKqkOf+SbVj/2fJJZ4RJMCfZp0rTGwRbzQD+HghfnhKOjL9E01okqz+ncJskGYfBNw==", "dev": true }, + "node_modules/uglify-js": { + "version": "3.17.4", + "resolved": "https://registry.npmjs.org/uglify-js/-/uglify-js-3.17.4.tgz", + "integrity": "sha512-T9q82TJI9e/C1TAxYvfb16xO120tMVFZrGA3f9/P4424DNu6ypK103y0GPFVa17yotwSyZW5iYXgjYHkGrJW/g==", + "optional": true, + "bin": { + "uglifyjs": "bin/uglifyjs" + }, + "engines": { + "node": ">=0.8.0" + } + }, "node_modules/unbox-primitive": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/unbox-primitive/-/unbox-primitive-1.0.2.tgz", @@ -9440,7 +9536,6 @@ "version": "2.0.1", "resolved": "https://registry.npmjs.org/universalify/-/universalify-2.0.1.tgz", "integrity": "sha512-gptHNQghINnc/vTGIk0SOFGFNXw7JVrlRUtConJRlvaw6DuX0wO5Jeko9sWrMBhh+PsYAZ7oXAiOnf/UKogyiw==", - "dev": true, "engines": { "node": ">= 10.0.0" } @@ -10391,6 +10486,11 @@ "node": ">=8" } }, + "node_modules/wordwrap": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/wordwrap/-/wordwrap-1.0.0.tgz", + "integrity": "sha512-gvVzJFlPycKc5dZN4yPkP8w7Dc37BtP1yczEneOb4uq34pXZcvrtRTmWV8W+Ume+XCxKgbjM+nevkyFPMybd4Q==" + }, "node_modules/wrap-ansi": { "version": "8.1.0", "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-8.1.0.tgz", diff --git a/package.json b/package.json index b50b7b396dab5f6a506c3e41c2984264c08acaef..872baf4d29ac7a1d14130784adcb27813b1871d0 100644 --- a/package.json +++ b/package.json @@ -22,6 +22,7 @@ "install": "^0.13.0", "js-cookie": "^3.0.5", "oh-vue-icons": "^1.0.0-rc3", + "openapi-typescript-codegen": "^0.29.0", "pinia": "^2.1.7", "pinia-plugin-persist": "^1.0.0", "pinia-plugin-persistedstate": "^3.2.1", diff --git a/spec.json b/spec.json index 39d296813341dac9fecbc619df7eb63d3a4287e9..68f66335ac507eb67b918d2924f79bb38aeedf5d 100644 --- a/spec.json +++ b/spec.json @@ -1 +1,1941 @@ -{"openapi":"3.0.1","info":{"title":"Sparesti API","description":"The Sparesti API","version":"3.0"},"servers":[{"url":"http://localhost:8080","description":"Generated server url"}],"security":[{"Bearer Authentication":[]}],"tags":[{"name":"Authentication","description":"User authentication"},{"name":"Leaderboard","description":"Retrieving leaderboard data"}],"paths":{"/api/auth/valid-email/{email}":{"post":{"tags":["Authentication"],"summary":"Validate email","description":"Check that the given email is valid","operationId":"validateEmail","parameters":[{"name":"email","in":"path","required":true,"schema":{"type":"string"}}],"responses":{"409":{"description":"Email already exists","content":{"*/*":{"schema":{"$ref":"#/components/schemas/ExceptionResponse"}}}},"200":{"description":"Email is valid","content":{"*/*":{"schema":{"type":"object"}}}}},"security":[]}},"/api/auth/signup":{"post":{"tags":["Authentication"],"summary":"User Signup","description":"Sign up a new user","operationId":"signup","requestBody":{"content":{"application/json":{"schema":{"$ref":"#/components/schemas/SignUpRequest"}}},"required":true},"responses":{"409":{"description":"Email already exists","content":{"application/json":{"schema":{"$ref":"#/components/schemas/ExceptionResponse"}}}},"201":{"description":"Successfully signed up","content":{"application/json":{"schema":{"$ref":"#/components/schemas/AuthenticationResponse"}}}}},"security":[]}},"/api/auth/login":{"post":{"tags":["Authentication"],"summary":"User Login","description":"Log in with an existing user","operationId":"login","requestBody":{"content":{"application/json":{"schema":{"$ref":"#/components/schemas/LoginRequest"}}},"required":true},"responses":{"200":{"description":"Successfully logged in","content":{"application/json":{"schema":{"$ref":"#/components/schemas/AuthenticationResponse"}}}},"401":{"description":"Invalid credentials","content":{"application/json":{"schema":{"$ref":"#/components/schemas/ExceptionResponse"}}}},"404":{"description":"User not found","content":{"application/json":{"schema":{"$ref":"#/components/schemas/ExceptionResponse"}}}}},"security":[]}},"/api/users":{"patch":{"tags":["user-controller"],"summary":"Update profile","description":"Update the profile of the authenticated user","operationId":"update","requestBody":{"content":{"application/json":{"schema":{"$ref":"#/components/schemas/UserUpdateDTO"}}},"required":true},"responses":{"200":{"description":"Successfully updated profile","content":{"application/json":{"schema":{"$ref":"#/components/schemas/UserDTO"}}}}}}},"/api/users/{userId}/profile":{"get":{"tags":["user-controller"],"summary":"Get profile","description":"Get user profile","operationId":"getProfile","parameters":[{"name":"userId","in":"path","required":true,"schema":{"type":"integer","format":"int64"}}],"responses":{"200":{"description":"Successfully got profile","content":{"application/json":{"schema":{"$ref":"#/components/schemas/ProfileDTO"}}}}}}},"/api/users/me":{"get":{"tags":["user-controller"],"summary":"Get user","description":"Get user information","operationId":"getUser","responses":{"200":{"description":"Successfully got user","content":{"application/json":{"schema":{"$ref":"#/components/schemas/UserDTO"}}}}}}},"/api/leaderboard":{"get":{"tags":["Leaderboard"],"operationId":"getLeaderboard","parameters":[{"name":"type","in":"query","required":true,"schema":{"type":"string"}},{"name":"filter","in":"query","required":true,"schema":{"type":"string"}},{"name":"entryCount","in":"query","required":false,"schema":{"type":"integer","format":"int32","default":10}}],"responses":{"200":{"description":"OK","content":{"application/json":{"schema":{"$ref":"#/components/schemas/LeaderboardDTO"}}}}}}},"/api/leaderboard/surrounding":{"get":{"tags":["Leaderboard"],"operationId":"getSurrounding","parameters":[{"name":"type","in":"query","required":true,"schema":{"type":"string"}},{"name":"filter","in":"query","required":true,"schema":{"type":"string"}},{"name":"entryCount","in":"query","required":false,"schema":{"type":"integer","format":"int32","default":10}}],"responses":{"200":{"description":"OK","content":{"application/json":{"schema":{"$ref":"#/components/schemas/LeaderboardDTO"}}}}}}}},"components":{"schemas":{"ExceptionResponse":{"type":"object","properties":{"status":{"type":"integer","format":"int32"},"message":{"type":"string"}}},"SignUpRequest":{"type":"object","properties":{"firstName":{"type":"string"},"lastName":{"type":"string"},"email":{"type":"string"},"password":{"type":"string"},"commitment":{"type":"string"},"experience":{"type":"string"},"challengeTypes":{"type":"array","items":{"type":"string"}}}},"AuthenticationResponse":{"type":"object","properties":{"firstName":{"type":"string"},"lastName":{"type":"string"},"role":{"type":"string"},"token":{"type":"string"}}},"LoginRequest":{"type":"object","properties":{"email":{"type":"string"},"password":{"type":"string"}}},"UserUpdateDTO":{"type":"object","properties":{"firstName":{"type":"string"},"lastName":{"type":"string"},"email":{"type":"string"},"password":{"type":"string"},"commitment":{"type":"string"},"experience":{"type":"string"},"challengeTypes":{"type":"array","items":{"type":"string"}}}},"UserDTO":{"type":"object","properties":{"id":{"type":"integer","format":"int64"},"firstName":{"type":"string"},"lastName":{"type":"string"},"email":{"type":"string"},"createdAt":{"type":"string","format":"date-time"},"role":{"type":"string"}}},"ProfileDTO":{"type":"object","properties":{"id":{"type":"integer","format":"int64"},"firstName":{"type":"string"},"lastName":{"type":"string"},"createdAt":{"type":"string","format":"date-time"}}},"LeaderboardDTO":{"type":"object","properties":{"type":{"type":"string"},"filter":{"type":"string"},"entries":{"type":"array","items":{"$ref":"#/components/schemas/LeaderboardEntryDTO"}}}},"LeaderboardEntryDTO":{"type":"object","properties":{"user":{"$ref":"#/components/schemas/UserDTO"},"score":{"type":"integer","format":"int32"},"rank":{"type":"integer","format":"int64"}}}},"securitySchemes":{"Bearer Authentication":{"type":"http","scheme":"bearer","bearerFormat":"JWT"}}}} \ No newline at end of file +{ + "openapi": "3.0.1", + "info": { + "title": "Sparesti API", + "description": "The Sparesti API", + "version": "3.0" + }, + "servers": [ + { + "url": "http://localhost:8080", + "description": "Generated server url" + } + ], + "security": [ + { + "Bearer Authentication": [] + } + ], + "paths": { + "/bank/v1/transaction/norwegian-domestic-payment-to-self": { + "post": { + "tags": [ + "transaction-controller" + ], + "summary": "Transfer to account", + "description": "Transfer money from a users account to another account of the same user", + "operationId": "transferToSelf", + "requestBody": { + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/TransactionDTO" + } + } + }, + "required": true + }, + "responses": { + "200": { + "description": "No accounts associated with a bank user", + "content": { + "*/*": { + "schema": { + "$ref": "#/components/schemas/TransactionDTO" + } + } + } + }, + "404": { + "description": "Bank profile id does not exist", + "content": { + "*/*": { + "schema": { + "$ref": "#/components/schemas/TransactionDTO" + } + } + } + } + } + } + }, + "/bank/v1/profile/create-profile": { + "post": { + "tags": [ + "bank-profile-controller" + ], + "summary": "Create bank profile", + "description": "Create a bank profile by providing a social security number", + "operationId": "createBankProfile", + "requestBody": { + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/BankProfileDTO" + } + } + }, + "required": true + }, + "responses": { + "200": { + "description": "Successfully created a bank profile", + "content": { + "*/*": { + "schema": { + "$ref": "#/components/schemas/BankProfileResponseDTO" + } + } + } + }, + "400": { + "description": "Could not create profile", + "content": { + "*/*": { + "schema": { + "$ref": "#/components/schemas/BankProfileResponseDTO" + } + } + } + } + } + } + }, + "/bank/v1/account/create-account": { + "post": { + "tags": [ + "account-controller" + ], + "summary": "Create account", + "description": "Create account with random balance", + "operationId": "createAccount", + "requestBody": { + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/AccountRequestDTO" + } + } + }, + "required": true + }, + "responses": { + "200": { + "description": "Successfully created account", + "content": { + "*/*": { + "schema": { + "$ref": "#/components/schemas/AccountResponseDTO" + } + } + } + }, + "404": { + "description": "Provided bank profile id could not be found", + "content": { + "*/*": { + "schema": { + "$ref": "#/components/schemas/AccountResponseDTO" + } + } + } + } + } + } + }, + "/api/users/reset-password": { + "post": { + "tags": [ + "User" + ], + "summary": "Initiate a password reset", + "description": "Send a password reset mail to the user with the specified email", + "operationId": "resetPassword", + "requestBody": { + "content": { + "text/plain": { + "schema": { + "type": "string" + } + } + }, + "required": true + }, + "responses": { + "202": { + "description": "Successfully initiated a password reset" + } + }, + "security": [] + } + }, + "/api/users/confirm-password": { + "post": { + "tags": [ + "User" + ], + "summary": "Confirm a password reset", + "description": "Confirms a password reset using a token and a new password", + "operationId": "confirmPasswordReset", + "requestBody": { + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/PasswordResetDTO" + } + } + }, + "required": true + }, + "responses": { + "204": { + "description": "Password was reset successfully" + }, + "403": { + "description": "Invalid token" + } + }, + "security": [] + } + }, + "/api/image/upload": { + "post": { + "tags": [ + "Image" + ], + "summary": "Upload an image", + "description": "Upload an image to the server", + "operationId": "uploadImage", + "requestBody": { + "content": { + "application/json": { + "schema": { + "required": [ + "file" + ], + "type": "object", + "properties": { + "file": { + "type": "string", + "format": "binary" + } + } + } + } + } + }, + "responses": { + "200": { + "description": "Successfully uploaded the image", + "content": { + "*/*": { + "schema": { + "type": "integer", + "format": "int64" + } + } + } + }, + "201": { + "description": "Created", + "content": { + "*/*": { + "schema": { + "type": "integer", + "format": "int64" + } + } + } + } + } + } + }, + "/api/goal/createGoal": { + "post": { + "tags": [ + "goal" + ], + "operationId": "createGoal", + "requestBody": { + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/CreateGoalDTO" + } + } + }, + "required": true + }, + "responses": { + "200": { + "description": "OK", + "content": { + "*/*": { + "schema": { + "$ref": "#/components/schemas/GoalDTO" + } + } + } + } + } + } + }, + "/api/budget/update": { + "post": { + "tags": [ + "User" + ], + "summary": "Updates a budget", + "description": "Updates a budget based on the budget request", + "operationId": "updateBudget", + "requestBody": { + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/BudgetResponseDTO" + } + } + }, + "required": true + }, + "responses": { + "200": { + "description": "Successfully updated budget", + "content": { + "application/json": { + "schema": { + "type": "object" + } + } + } + }, + "500": { + "description": "Budget is not found", + "content": { + "application/json": { + "schema": { + "type": "object" + } + } + } + } + } + } + }, + "/api/budget/update/expense/{budgetId}": { + "post": { + "tags": [ + "User" + ], + "summary": "Created/Updates an expense", + "description": "Creates/Updates a budget based on the budget request", + "operationId": "updateExpense", + "parameters": [ + { + "name": "budgetId", + "in": "path", + "required": true, + "schema": { + "type": "integer", + "format": "int64" + } + } + ], + "requestBody": { + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/ExpenseRequestDTO" + } + } + }, + "required": true + }, + "responses": { + "200": { + "description": "Successfully updated budget", + "content": { + "application/json": { + "schema": { + "type": "object" + } + } + } + }, + "500": { + "description": "Error updating expense", + "content": { + "application/json": { + "schema": { + "type": "object" + } + } + } + } + } + } + }, + "/api/budget/create": { + "post": { + "tags": [ + "User" + ], + "summary": "Create a new budget", + "description": "Create a new budget with based on the budget request", + "operationId": "createBudget", + "requestBody": { + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/BudgetRequestDTO" + } + } + }, + "required": true + }, + "responses": { + "200": { + "description": "Successfully created new budget", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/Budget" + } + } + } + } + } + } + }, + "/api/auth/valid-email/{email}": { + "post": { + "tags": [ + "Authentication" + ], + "summary": "Validate email", + "description": "Check that the given email is valid", + "operationId": "validateEmail", + "parameters": [ + { + "name": "email", + "in": "path", + "required": true, + "schema": { + "type": "string" + } + } + ], + "responses": { + "200": { + "description": "Email is valid", + "content": { + "*/*": { + "schema": { + "type": "object" + } + } + } + }, + "409": { + "description": "Email already exists", + "content": { + "*/*": { + "schema": { + "$ref": "#/components/schemas/ExceptionResponse" + } + } + } + } + }, + "security": [] + } + }, + "/api/auth/signup": { + "post": { + "tags": [ + "Authentication" + ], + "summary": "User Signup", + "description": "Sign up a new user", + "operationId": "signup", + "requestBody": { + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/SignUpRequest" + } + } + }, + "required": true + }, + "responses": { + "201": { + "description": "Successfully signed up", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/AuthenticationResponse" + } + } + } + }, + "409": { + "description": "Email already exists", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/ExceptionResponse" + } + } + } + } + }, + "security": [] + } + }, + "/api/auth/login": { + "post": { + "tags": [ + "Authentication" + ], + "summary": "User Login", + "description": "Log in with an existing user", + "operationId": "login", + "requestBody": { + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/LoginRequest" + } + } + }, + "required": true + }, + "responses": { + "200": { + "description": "Successfully logged in", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/AuthenticationResponse" + } + } + } + }, + "401": { + "description": "Invalid credentials", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/ExceptionResponse" + } + } + } + }, + "404": { + "description": "User not found", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/ExceptionResponse" + } + } + } + } + }, + "security": [] + } + }, + "/api/users": { + "patch": { + "tags": [ + "User" + ], + "summary": "Update a profile", + "description": "Update the profile of the authenticated user", + "operationId": "update", + "requestBody": { + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/UserUpdateDTO" + } + } + }, + "required": true + }, + "responses": { + "200": { + "description": "Successfully updated profile", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/UserDTO" + } + } + } + } + } + } + }, + "/api/users/update-account": { + "patch": { + "tags": [ + "User" + ], + "summary": "Update a user's bank account", + "description": "Changes either a user's checking account or savings account", + "operationId": "selectBankAccount", + "requestBody": { + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/BankAccountDTO" + } + } + }, + "required": true + }, + "responses": { + "200": { + "description": "OK", + "content": { + "*/*": { + "schema": { + "$ref": "#/components/schemas/Account" + } + } + } + } + } + } + }, + "/bank/v1/account/accounts/ssn/{ssn}": { + "get": { + "tags": [ + "account-controller" + ], + "summary": "Get user accounts", + "description": "Get accounts associated with a user by providing their social security number", + "operationId": "getAccountsBySsn", + "parameters": [ + { + "name": "ssn", + "in": "path", + "required": true, + "schema": { + "type": "integer", + "format": "int64" + } + } + ], + "responses": { + "200": { + "description": "No accounts associated with a bank user", + "content": { + "*/*": { + "schema": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Account" + } + } + } + } + }, + "404": { + "description": "Social security number does not exist", + "content": { + "*/*": { + "schema": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Account" + } + } + } + } + } + } + } + }, + "/bank/v1/account/accounts/profile/{bankProfileId}": { + "get": { + "tags": [ + "account-controller" + ], + "summary": "Get user accounts", + "description": "Get accounts associated with a user by providing their bank profile id", + "operationId": "getAccounts", + "parameters": [ + { + "name": "bankProfileId", + "in": "path", + "required": true, + "schema": { + "type": "integer", + "format": "int64" + } + } + ], + "responses": { + "200": { + "description": "No accounts associated with a bank user", + "content": { + "*/*": { + "schema": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Account" + } + } + } + } + }, + "404": { + "description": "Bank profile id does not exist", + "content": { + "*/*": { + "schema": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Account" + } + } + } + } + } + } + } + }, + "/api/users/{userId}/profile": { + "get": { + "tags": [ + "User" + ], + "summary": "Get a profile", + "description": "Get the profile of a user", + "operationId": "getProfile", + "parameters": [ + { + "name": "userId", + "in": "path", + "required": true, + "schema": { + "type": "integer", + "format": "int64" + } + } + ], + "responses": { + "200": { + "description": "Successfully got profile", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/ProfileDTO" + } + } + } + } + } + } + }, + "/api/users/me": { + "get": { + "tags": [ + "User" + ], + "summary": "Get the authenticated user", + "description": "Get all user information for the authenticated user", + "operationId": "getUser", + "responses": { + "200": { + "description": "Successfully got user", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/UserDTO" + } + } + } + } + } + } + }, + "/api/leaderboard": { + "get": { + "tags": [ + "Leaderboard" + ], + "operationId": "getLeaderboard", + "parameters": [ + { + "name": "type", + "in": "query", + "required": true, + "schema": { + "type": "string" + } + }, + { + "name": "filter", + "in": "query", + "required": true, + "schema": { + "type": "string" + } + }, + { + "name": "entryCount", + "in": "query", + "required": false, + "schema": { + "type": "integer", + "format": "int32", + "default": 10 + } + } + ], + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/LeaderboardDTO" + } + } + } + } + } + } + }, + "/api/leaderboard/surrounding": { + "get": { + "tags": [ + "Leaderboard" + ], + "operationId": "getSurrounding", + "parameters": [ + { + "name": "type", + "in": "query", + "required": true, + "schema": { + "type": "string" + } + }, + { + "name": "filter", + "in": "query", + "required": true, + "schema": { + "type": "string" + } + }, + { + "name": "entryCount", + "in": "query", + "required": false, + "schema": { + "type": "integer", + "format": "int32", + "default": 10 + } + } + ], + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/LeaderboardDTO" + } + } + } + } + } + } + }, + "/api/image/{id}": { + "get": { + "tags": [ + "Image" + ], + "summary": "Retrieve an image", + "description": "Retrieve an image from the server", + "operationId": "getImage", + "parameters": [ + { + "name": "id", + "in": "path", + "required": true, + "schema": { + "type": "integer", + "format": "int64" + } + } + ], + "responses": { + "200": { + "description": "Successfully retrieved the image", + "content": { + "*/*": { + "schema": { + "type": "array", + "items": { + "type": "string", + "format": "byte" + } + } + } + } + } + } + } + }, + "/api/goal/getGoals": { + "get": { + "tags": [ + "goal" + ], + "operationId": "getGoals", + "responses": { + "200": { + "description": "OK", + "content": { + "*/*": { + "schema": { + "type": "array", + "items": { + "$ref": "#/components/schemas/GoalDTO" + } + } + } + } + } + } + } + }, + "/api/goal/getGoal": { + "get": { + "tags": [ + "goal" + ], + "operationId": "getGoal", + "responses": { + "200": { + "description": "OK", + "content": { + "*/*": { + "schema": { + "$ref": "#/components/schemas/GoalDTO" + } + } + } + } + } + } + }, + "/api/budget": { + "get": { + "tags": [ + "User" + ], + "summary": "Get the list of budgets", + "description": "Get all budgets related to the authenticated user", + "operationId": "getBudgetsByUser", + "responses": { + "200": { + "description": "Successfully got budgets", + "content": { + "application/json": { + "schema": { + "type": "array", + "items": { + "$ref": "#/components/schemas/BudgetResponseDTO" + } + } + } + } + } + } + } + }, + "/api/budget/{budgetId}": { + "get": { + "tags": [ + "User" + ], + "summary": "Get the budget", + "description": "Get budget by its id ", + "operationId": "getBudget", + "parameters": [ + { + "name": "budgetId", + "in": "path", + "required": true, + "schema": { + "type": "integer", + "format": "int64" + } + } + ], + "responses": { + "200": { + "description": "Successfully got budget", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/BudgetResponseDTO" + } + } + } + }, + "500": { + "description": "Budget is not found", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/BudgetResponseDTO" + } + } + } + } + } + } + }, + "/api/budget/expenses/{budgetId}": { + "get": { + "tags": [ + "User" + ], + "summary": "Get the list of budgets", + "description": "Get all budgets related to the authenticated user", + "operationId": "getExpenses", + "parameters": [ + { + "name": "budgetId", + "in": "path", + "required": true, + "schema": { + "type": "integer", + "format": "int64" + } + } + ], + "responses": { + "200": { + "description": "Successfully got expenses", + "content": { + "application/json": { + "schema": { + "type": "array", + "items": { + "$ref": "#/components/schemas/ExpenseResponseDTO" + } + } + } + } + } + } + } + }, + "/api/budget/expense/{expenseId}": { + "get": { + "tags": [ + "User" + ], + "summary": "Get the expense", + "description": "Get expense by its id ", + "operationId": "getExpense", + "parameters": [ + { + "name": "expenseId", + "in": "path", + "required": true, + "schema": { + "type": "integer", + "format": "int64" + } + } + ], + "responses": { + "200": { + "description": "Successfully got expense", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/ExpenseResponseDTO" + } + } + } + }, + "500": { + "description": "Expense is not found", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/ExpenseResponseDTO" + } + } + } + } + } + } + }, + "/api/budget/delete/{budgetId}": { + "get": { + "tags": [ + "User" + ], + "summary": "Deletes a budget", + "description": "Deletes a budget based on provided budget id", + "operationId": "deleteBudget", + "parameters": [ + { + "name": "budgetId", + "in": "path", + "required": true, + "schema": { + "type": "integer", + "format": "int64" + } + } + ], + "responses": { + "200": { + "description": "Successfully deleted budget", + "content": { + "application/json": { + "schema": { + "type": "object" + } + } + } + }, + "500": { + "description": "Budget is not found", + "content": { + "application/json": { + "schema": { + "type": "object" + } + } + } + } + } + } + }, + "/api/budget/delete/expense/{expenseId}": { + "get": { + "tags": [ + "User" + ], + "summary": "Deletes an expense", + "description": "Deletes an expense based on provided expense id", + "operationId": "deleteExpense", + "parameters": [ + { + "name": "expenseId", + "in": "path", + "required": true, + "schema": { + "type": "integer", + "format": "int64" + } + } + ], + "responses": { + "200": { + "description": "Successfully deleted expense", + "content": { + "application/json": { + "schema": { + "type": "object" + } + } + } + }, + "500": { + "description": "Expense is not found", + "content": { + "application/json": { + "schema": { + "type": "object" + } + } + } + } + } + } + } + }, + "components": { + "schemas": { + "TransactionDTO": { + "type": "object", + "properties": { + "debtorBBAN": { + "type": "integer", + "format": "int64" + }, + "creditorBBAN": { + "type": "integer", + "format": "int64" + }, + "amount": { + "type": "number" + } + } + }, + "BankProfileDTO": { + "type": "object", + "properties": { + "ssn": { + "type": "integer", + "format": "int64" + } + } + }, + "Account": { + "type": "object", + "properties": { + "bban": { + "type": "integer", + "format": "int64" + }, + "balance": { + "type": "number" + }, + "bankProfile": { + "$ref": "#/components/schemas/BankProfile" + } + } + }, + "BankProfile": { + "type": "object", + "properties": { + "id": { + "type": "integer", + "format": "int64" + }, + "ssn": { + "type": "integer", + "format": "int64" + }, + "accounts": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Account" + } + } + } + }, + "BankProfileResponseDTO": { + "type": "object", + "properties": { + "ssn": { + "type": "integer", + "format": "int64" + }, + "accounts": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Account" + } + } + } + }, + "AccountRequestDTO": { + "type": "object", + "properties": { + "ssn": { + "type": "integer", + "format": "int64" + } + } + }, + "AccountResponseDTO": { + "type": "object", + "properties": { + "bankProfileId": { + "type": "integer", + "format": "int64" + }, + "balance": { + "type": "number" + } + } + }, + "PasswordResetDTO": { + "required": [ + "token" + ], + "type": "object", + "properties": { + "token": { + "type": "string" + }, + "password": { + "type": "string" + } + } + }, + "CreateGoalDTO": { + "type": "object", + "properties": { + "goalName": { + "type": "string" + }, + "description": { + "type": "string" + }, + "targetAmount": { + "type": "integer", + "format": "int32" + }, + "targetDate": { + "type": "string", + "format": "date-time" + } + } + }, + "ChallengeDTO": { + "type": "object", + "properties": { + "id": { + "type": "integer", + "format": "int64" + }, + "potentialSavingAmount": { + "type": "integer", + "format": "int32" + }, + "points": { + "type": "integer", + "format": "int32" + }, + "days": { + "type": "integer", + "format": "int32" + }, + "createdAt": { + "type": "string", + "format": "date-time" + }, + "dailyChallengeProgressList": { + "type": "array", + "items": { + "$ref": "#/components/schemas/DailyChallengeProgressDTO" + } + } + } + }, + "DailyChallengeProgressDTO": { + "type": "object", + "properties": { + "id": { + "type": "integer", + "format": "int64" + }, + "challengeDay": { + "type": "integer", + "format": "int32" + }, + "completedAt": { + "type": "string", + "format": "date-time" + } + } + }, + "GoalDTO": { + "type": "object", + "properties": { + "id": { + "type": "integer", + "format": "int64" + }, + "goalName": { + "type": "string" + }, + "description": { + "type": "string" + }, + "targetAmount": { + "type": "integer", + "format": "int32" + }, + "targetDate": { + "type": "string", + "format": "date-time" + }, + "completedAt": { + "type": "string", + "format": "date-time" + }, + "createdAt": { + "type": "string", + "format": "date-time" + }, + "challenges": { + "type": "array", + "items": { + "$ref": "#/components/schemas/ChallengeDTO" + } + }, + "participants": { + "type": "array", + "items": { + "$ref": "#/components/schemas/ParticipantDTO" + } + } + } + }, + "ParticipantDTO": { + "type": "object", + "properties": { + "role": { + "type": "string", + "enum": [ + "CREATOR", + "CONTRIBUTOR" + ] + }, + "user": { + "$ref": "#/components/schemas/ParticipantUserDTO" + } + } + }, + "ParticipantUserDTO": { + "type": "object", + "properties": { + "firstName": { + "type": "string" + }, + "lastName": { + "type": "string" + } + } + }, + "BudgetResponseDTO": { + "type": "object", + "properties": { + "id": { + "type": "integer", + "format": "int64" + }, + "budgetName": { + "type": "string" + }, + "budgetAmount": { + "type": "number" + }, + "expenseAmount": { + "type": "number" + }, + "createdAt": { + "type": "string", + "format": "date-time" + } + } + }, + "ExpenseRequestDTO": { + "type": "object", + "properties": { + "expenseId": { + "type": "integer", + "format": "int64" + }, + "description": { + "type": "string" + }, + "amount": { + "type": "number" + } + } + }, + "BudgetRequestDTO": { + "type": "object", + "properties": { + "budgetName": { + "type": "string" + }, + "budgetAmount": { + "type": "number" + }, + "expenseAmount": { + "type": "number" + } + } + }, + "Badge": { + "type": "object", + "properties": { + "id": { + "type": "integer", + "format": "int64" + }, + "badgeName": { + "type": "string" + }, + "criteria": { + "type": "string" + }, + "badgeUserList": { + "type": "array", + "items": { + "$ref": "#/components/schemas/BadgeUser" + } + } + } + }, + "BadgeUser": { + "type": "object", + "properties": { + "badgeUserId": { + "$ref": "#/components/schemas/BadgeUserId" + }, + "earnedAt": { + "type": "string", + "format": "date-time" + } + } + }, + "BadgeUserId": { + "type": "object", + "properties": { + "badge": { + "$ref": "#/components/schemas/Badge" + }, + "user": { + "$ref": "#/components/schemas/User" + } + } + }, + "Budget": { + "type": "object", + "properties": { + "id": { + "type": "integer", + "format": "int64" + }, + "user": { + "$ref": "#/components/schemas/User" + }, + "createdAt": { + "type": "string", + "format": "date-time" + }, + "budgetName": { + "type": "string" + }, + "budgetAmount": { + "type": "number" + }, + "expenseAmount": { + "type": "number" + } + } + }, + "Configuration": { + "type": "object", + "properties": { + "id": { + "type": "integer", + "format": "int64" + }, + "commitment": { + "type": "string", + "enum": [ + "LITTLE", + "SOME", + "MUCH" + ] + }, + "experience": { + "type": "string", + "enum": [ + "NONE", + "SOME", + "EXPERT" + ] + }, + "challengeTypes": { + "type": "array", + "items": { + "type": "string", + "enum": [ + "NO_COFFEE", + "NO_CAR", + "SHORTER_SHOWER", + "SPEND_LESS_ON_FOOD", + "BUY_USED_CLOTHES", + "LESS_SHOPPING", + "DROP_SUBSCRIPTION", + "SELL_SOMETHING", + "BUY_USED", + "EAT_PACKED_LUNCH", + "STOP_SHOPPING", + "ZERO_SPENDING", + "RENT_YOUR_STUFF", + "MEATLESS", + "SCREEN_TIME_LIMIT", + "UNPLUGGED_ENTERTAINMENT" + ] + } + } + } + }, + "GrantedAuthority": { + "type": "object", + "properties": { + "authority": { + "type": "string" + } + } + }, + "Point": { + "type": "object", + "properties": { + "id": { + "type": "integer", + "format": "int64" + }, + "currentPoints": { + "type": "integer", + "format": "int32" + }, + "totalEarnedPoints": { + "type": "integer", + "format": "int32" + } + } + }, + "Streak": { + "type": "object", + "properties": { + "id": { + "type": "integer", + "format": "int64" + }, + "currentStreak": { + "type": "integer", + "format": "int32" + }, + "currentStreakCreatedAt": { + "type": "string", + "format": "date-time" + }, + "currentStreakUpdatedAt": { + "type": "string", + "format": "date-time" + }, + "highestStreak": { + "type": "integer", + "format": "int32" + }, + "highestStreakCreatedAt": { + "type": "string", + "format": "date-time" + }, + "highestStreakEndedAt": { + "type": "string", + "format": "date-time" + } + } + }, + "User": { + "type": "object", + "properties": { + "id": { + "type": "integer", + "format": "int64" + }, + "firstName": { + "type": "string" + }, + "lastName": { + "type": "string" + }, + "email": { + "type": "string" + }, + "checkingAccount": { + "$ref": "#/components/schemas/Account" + }, + "savingsAccount": { + "$ref": "#/components/schemas/Account" + }, + "password": { + "type": "string" + }, + "createdAt": { + "type": "string", + "format": "date-time" + }, + "role": { + "type": "string", + "enum": [ + "USER", + "ADMIN" + ] + }, + "badges": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Badge" + } + }, + "point": { + "$ref": "#/components/schemas/Point" + }, + "streak": { + "$ref": "#/components/schemas/Streak" + }, + "configuration": { + "$ref": "#/components/schemas/Configuration" + }, + "enabled": { + "type": "boolean" + }, + "accountNonExpired": { + "type": "boolean" + }, + "credentialsNonExpired": { + "type": "boolean" + }, + "username": { + "type": "string" + }, + "authorities": { + "type": "array", + "items": { + "$ref": "#/components/schemas/GrantedAuthority" + } + }, + "accountNonLocked": { + "type": "boolean" + } + } + }, + "ExceptionResponse": { + "type": "object", + "properties": { + "status": { + "type": "integer", + "format": "int32" + }, + "message": { + "type": "string" + } + } + }, + "ConfigurationDTO": { + "type": "object", + "properties": { + "commitment": { + "type": "string" + }, + "experience": { + "type": "string" + }, + "challengeTypes": { + "type": "array", + "items": { + "type": "string" + } + } + } + }, + "SignUpRequest": { + "required": [ + "configuration" + ], + "type": "object", + "properties": { + "firstName": { + "type": "string" + }, + "lastName": { + "type": "string" + }, + "email": { + "type": "string" + }, + "password": { + "type": "string" + }, + "configuration": { + "$ref": "#/components/schemas/ConfigurationDTO" + } + } + }, + "AuthenticationResponse": { + "type": "object", + "properties": { + "firstName": { + "type": "string" + }, + "lastName": { + "type": "string" + }, + "role": { + "type": "string" + }, + "token": { + "type": "string" + } + } + }, + "LoginRequest": { + "type": "object", + "properties": { + "email": { + "type": "string" + }, + "password": { + "type": "string" + } + } + }, + "UserUpdateDTO": { + "type": "object", + "properties": { + "firstName": { + "type": "string" + }, + "lastName": { + "type": "string" + }, + "email": { + "type": "string" + }, + "password": { + "type": "string" + }, + "configuration": { + "$ref": "#/components/schemas/ConfigurationDTO" + } + } + }, + "UserDTO": { + "type": "object", + "properties": { + "id": { + "type": "integer", + "format": "int64" + }, + "firstName": { + "type": "string" + }, + "lastName": { + "type": "string" + }, + "email": { + "type": "string" + }, + "createdAt": { + "type": "string", + "format": "date-time" + }, + "role": { + "type": "string" + } + } + }, + "BankAccountDTO": { + "type": "object", + "properties": { + "bban": { + "type": "integer", + "format": "int64" + }, + "bankAccountType": { + "type": "string" + } + } + }, + "ProfileDTO": { + "type": "object", + "properties": { + "id": { + "type": "integer", + "format": "int64" + }, + "firstName": { + "type": "string" + }, + "lastName": { + "type": "string" + }, + "createdAt": { + "type": "string", + "format": "date-time" + } + } + }, + "LeaderboardDTO": { + "type": "object", + "properties": { + "type": { + "type": "string" + }, + "filter": { + "type": "string" + }, + "entries": { + "type": "array", + "items": { + "$ref": "#/components/schemas/LeaderboardEntryDTO" + } + } + } + }, + "LeaderboardEntryDTO": { + "type": "object", + "properties": { + "user": { + "$ref": "#/components/schemas/UserDTO" + }, + "score": { + "type": "integer", + "format": "int32" + }, + "rank": { + "type": "integer", + "format": "int64" + } + } + }, + "ExpenseResponseDTO": { + "type": "object", + "properties": { + "expenseId": { + "type": "integer", + "format": "int64" + }, + "budgetId": { + "type": "integer", + "format": "int64" + }, + "description": { + "type": "string" + }, + "amount": { + "type": "string" + } + } + } + }, + "securitySchemes": { + "Bearer Authentication": { + "type": "http", + "scheme": "bearer", + "bearerFormat": "JWT" + } + } + } +} \ No newline at end of file diff --git a/src/api/core/request.ts b/src/api/core/request.ts index c6716ef792719600ab424d980a6af2796f53fcd5..1dc6fef4aab4086ff57b48d26b20ec26bb8fa472 100644 --- a/src/api/core/request.ts +++ b/src/api/core/request.ts @@ -4,7 +4,6 @@ /* eslint-disable */ import axios from 'axios'; import type { AxiosError, AxiosRequestConfig, AxiosResponse, AxiosInstance } from 'axios'; -//@ts-ignore import FormData from 'form-data'; import { ApiError } from './ApiError'; diff --git a/src/api/index.ts b/src/api/index.ts index b63858683d0644088eaa29f307035f0f4d44ef0b..a3a2de06404f9a80b4f14a9e08e8857581203498 100644 --- a/src/api/index.ts +++ b/src/api/index.ts @@ -11,24 +11,38 @@ export type { Account } from './models/Account'; export type { AccountRequestDTO } from './models/AccountRequestDTO'; export type { AccountResponseDTO } from './models/AccountResponseDTO'; export type { AuthenticationResponse } from './models/AuthenticationResponse'; +export type { Badge } from './models/Badge'; +export type { BadgeUser } from './models/BadgeUser'; +export type { BadgeUserId } from './models/BadgeUserId'; +export type { BankAccountDTO } from './models/BankAccountDTO'; export type { BankProfile } from './models/BankProfile'; export type { BankProfileDTO } from './models/BankProfileDTO'; export type { BankProfileResponseDTO } from './models/BankProfileResponseDTO'; +export type { Budget } from './models/Budget'; +export type { BudgetRequestDTO } from './models/BudgetRequestDTO'; +export type { BudgetResponseDTO } from './models/BudgetResponseDTO'; export type { ChallengeDTO } from './models/ChallengeDTO'; +export { Configuration } from './models/Configuration'; export type { ConfigurationDTO } from './models/ConfigurationDTO'; export type { CreateGoalDTO } from './models/CreateGoalDTO'; export type { DailyChallengeProgressDTO } from './models/DailyChallengeProgressDTO'; export type { ExceptionResponse } from './models/ExceptionResponse'; +export type { ExpenseRequestDTO } from './models/ExpenseRequestDTO'; +export type { ExpenseResponseDTO } from './models/ExpenseResponseDTO'; export type { GoalDTO } from './models/GoalDTO'; +export type { GrantedAuthority } from './models/GrantedAuthority'; export type { LeaderboardDTO } from './models/LeaderboardDTO'; export type { LeaderboardEntryDTO } from './models/LeaderboardEntryDTO'; export type { LoginRequest } from './models/LoginRequest'; export { ParticipantDTO } from './models/ParticipantDTO'; export type { ParticipantUserDTO } from './models/ParticipantUserDTO'; export type { PasswordResetDTO } from './models/PasswordResetDTO'; +export type { Point } from './models/Point'; export type { ProfileDTO } from './models/ProfileDTO'; export type { SignUpRequest } from './models/SignUpRequest'; +export type { Streak } from './models/Streak'; export type { TransactionDTO } from './models/TransactionDTO'; +export { User } from './models/User'; export type { UserDTO } from './models/UserDTO'; export type { UserUpdateDTO } from './models/UserUpdateDTO'; @@ -36,6 +50,7 @@ export { AccountControllerService } from './services/AccountControllerService'; export { AuthenticationService } from './services/AuthenticationService'; export { BankProfileControllerService } from './services/BankProfileControllerService'; export { GoalService } from './services/GoalService'; +export { ImageService } from './services/ImageService'; export { LeaderboardService } from './services/LeaderboardService'; export { TransactionControllerService } from './services/TransactionControllerService'; export { UserService } from './services/UserService'; diff --git a/src/api/models/Badge.ts b/src/api/models/Badge.ts new file mode 100644 index 0000000000000000000000000000000000000000..50014a6bafe26372c8a1de0590202b0d0fdd9a13 --- /dev/null +++ b/src/api/models/Badge.ts @@ -0,0 +1,12 @@ +/* generated using openapi-typescript-codegen -- do not edit */ +/* istanbul ignore file */ +/* tslint:disable */ +/* eslint-disable */ +import type { BadgeUser } from './BadgeUser'; +export type Badge = { + id?: number; + badgeName?: string; + criteria?: string; + badgeUserList?: Array<BadgeUser>; +}; + diff --git a/src/api/models/BadgeUser.ts b/src/api/models/BadgeUser.ts new file mode 100644 index 0000000000000000000000000000000000000000..46d6ec47b928849f1765ef5e0a15cca35567cfbb --- /dev/null +++ b/src/api/models/BadgeUser.ts @@ -0,0 +1,10 @@ +/* generated using openapi-typescript-codegen -- do not edit */ +/* istanbul ignore file */ +/* tslint:disable */ +/* eslint-disable */ +import type { BadgeUserId } from './BadgeUserId'; +export type BadgeUser = { + badgeUserId?: BadgeUserId; + earnedAt?: string; +}; + diff --git a/src/api/models/BadgeUserId.ts b/src/api/models/BadgeUserId.ts new file mode 100644 index 0000000000000000000000000000000000000000..d89476aa54dcae9fad5ba96e9e7a6e2be2eb1c71 --- /dev/null +++ b/src/api/models/BadgeUserId.ts @@ -0,0 +1,11 @@ +/* generated using openapi-typescript-codegen -- do not edit */ +/* istanbul ignore file */ +/* tslint:disable */ +/* eslint-disable */ +import type { Badge } from './Badge'; +import type { User } from './User'; +export type BadgeUserId = { + badge?: Badge; + user?: User; +}; + diff --git a/src/api/models/BankAccountDTO.ts b/src/api/models/BankAccountDTO.ts new file mode 100644 index 0000000000000000000000000000000000000000..b1ac2aac884d14d4e4a3241482e6861a8e91a7ab --- /dev/null +++ b/src/api/models/BankAccountDTO.ts @@ -0,0 +1,9 @@ +/* generated using openapi-typescript-codegen -- do not edit */ +/* istanbul ignore file */ +/* tslint:disable */ +/* eslint-disable */ +export type BankAccountDTO = { + bban?: number; + bankAccountType?: string; +}; + diff --git a/src/api/models/Budget.ts b/src/api/models/Budget.ts new file mode 100644 index 0000000000000000000000000000000000000000..6103ac46760fb282236f05a0ba77eb3a5031cf2f --- /dev/null +++ b/src/api/models/Budget.ts @@ -0,0 +1,14 @@ +/* generated using openapi-typescript-codegen -- do not edit */ +/* istanbul ignore file */ +/* tslint:disable */ +/* eslint-disable */ +import type { User } from './User'; +export type Budget = { + id?: number; + user?: User; + createdAt?: string; + budgetName?: string; + budgetAmount?: number; + expenseAmount?: number; +}; + diff --git a/src/api/models/BudgetRequestDTO.ts b/src/api/models/BudgetRequestDTO.ts new file mode 100644 index 0000000000000000000000000000000000000000..d4800b3b9b4ad2d2f2ce04cc6ea2d2bedf1cd76e --- /dev/null +++ b/src/api/models/BudgetRequestDTO.ts @@ -0,0 +1,10 @@ +/* generated using openapi-typescript-codegen -- do not edit */ +/* istanbul ignore file */ +/* tslint:disable */ +/* eslint-disable */ +export type BudgetRequestDTO = { + budgetName?: string; + budgetAmount?: number; + expenseAmount?: number; +}; + diff --git a/src/api/models/BudgetResponseDTO.ts b/src/api/models/BudgetResponseDTO.ts new file mode 100644 index 0000000000000000000000000000000000000000..04a579845fbac1129b9d23bde7d31ed661ee5465 --- /dev/null +++ b/src/api/models/BudgetResponseDTO.ts @@ -0,0 +1,12 @@ +/* generated using openapi-typescript-codegen -- do not edit */ +/* istanbul ignore file */ +/* tslint:disable */ +/* eslint-disable */ +export type BudgetResponseDTO = { + id?: number; + budgetName?: string; + budgetAmount?: number; + expenseAmount?: number; + createdAt?: string; +}; + diff --git a/src/api/models/Configuration.ts b/src/api/models/Configuration.ts new file mode 100644 index 0000000000000000000000000000000000000000..b932eb0ee7940f855a894f6b0117751b4d3e0156 --- /dev/null +++ b/src/api/models/Configuration.ts @@ -0,0 +1,23 @@ +/* generated using openapi-typescript-codegen -- do not edit */ +/* istanbul ignore file */ +/* tslint:disable */ +/* eslint-disable */ +export type Configuration = { + id?: number; + commitment?: Configuration.commitment; + experience?: Configuration.experience; + challengeTypes?: Array<'NO_COFFEE' | 'NO_CAR' | 'SHORTER_SHOWER' | 'SPEND_LESS_ON_FOOD' | 'BUY_USED_CLOTHES' | 'LESS_SHOPPING' | 'DROP_SUBSCRIPTION' | 'SELL_SOMETHING' | 'BUY_USED' | 'EAT_PACKED_LUNCH' | 'STOP_SHOPPING' | 'ZERO_SPENDING' | 'RENT_YOUR_STUFF' | 'MEATLESS' | 'SCREEN_TIME_LIMIT' | 'UNPLUGGED_ENTERTAINMENT'>; +}; +export namespace Configuration { + export enum commitment { + LITTLE = 'LITTLE', + SOME = 'SOME', + MUCH = 'MUCH', + } + export enum experience { + NONE = 'NONE', + SOME = 'SOME', + EXPERT = 'EXPERT', + } +} + diff --git a/src/api/models/ExpenseRequestDTO.ts b/src/api/models/ExpenseRequestDTO.ts new file mode 100644 index 0000000000000000000000000000000000000000..c4337af1a3b5617e25cff23b80f256957d9fc22b --- /dev/null +++ b/src/api/models/ExpenseRequestDTO.ts @@ -0,0 +1,10 @@ +/* generated using openapi-typescript-codegen -- do not edit */ +/* istanbul ignore file */ +/* tslint:disable */ +/* eslint-disable */ +export type ExpenseRequestDTO = { + expenseId?: number; + description?: string; + amount?: number; +}; + diff --git a/src/api/models/ExpenseResponseDTO.ts b/src/api/models/ExpenseResponseDTO.ts new file mode 100644 index 0000000000000000000000000000000000000000..35bcc0e0b2bc582aa8fb5c9bcd7756d9a213d1ce --- /dev/null +++ b/src/api/models/ExpenseResponseDTO.ts @@ -0,0 +1,11 @@ +/* generated using openapi-typescript-codegen -- do not edit */ +/* istanbul ignore file */ +/* tslint:disable */ +/* eslint-disable */ +export type ExpenseResponseDTO = { + expenseId?: number; + budgetId?: number; + description?: string; + amount?: string; +}; + diff --git a/src/api/models/GrantedAuthority.ts b/src/api/models/GrantedAuthority.ts new file mode 100644 index 0000000000000000000000000000000000000000..5fa268a0a717b478a8b6abcf5ccf8ccd2ecf5ae8 --- /dev/null +++ b/src/api/models/GrantedAuthority.ts @@ -0,0 +1,8 @@ +/* generated using openapi-typescript-codegen -- do not edit */ +/* istanbul ignore file */ +/* tslint:disable */ +/* eslint-disable */ +export type GrantedAuthority = { + authority?: string; +}; + diff --git a/src/api/models/Point.ts b/src/api/models/Point.ts new file mode 100644 index 0000000000000000000000000000000000000000..971f5ff1cf23bbe7dff322dc589cafc38706497d --- /dev/null +++ b/src/api/models/Point.ts @@ -0,0 +1,10 @@ +/* generated using openapi-typescript-codegen -- do not edit */ +/* istanbul ignore file */ +/* tslint:disable */ +/* eslint-disable */ +export type Point = { + id?: number; + currentPoints?: number; + totalEarnedPoints?: number; +}; + diff --git a/src/api/models/Streak.ts b/src/api/models/Streak.ts new file mode 100644 index 0000000000000000000000000000000000000000..d228c48e7a51d763f5ddcbd86d2143144c36b60a --- /dev/null +++ b/src/api/models/Streak.ts @@ -0,0 +1,14 @@ +/* generated using openapi-typescript-codegen -- do not edit */ +/* istanbul ignore file */ +/* tslint:disable */ +/* eslint-disable */ +export type Streak = { + id?: number; + currentStreak?: number; + currentStreakCreatedAt?: string; + currentStreakUpdatedAt?: string; + highestStreak?: number; + highestStreakCreatedAt?: string; + highestStreakEndedAt?: string; +}; + diff --git a/src/api/models/User.ts b/src/api/models/User.ts new file mode 100644 index 0000000000000000000000000000000000000000..0eaf2816d9ef7030fe434d45344fff6068a9e4a5 --- /dev/null +++ b/src/api/models/User.ts @@ -0,0 +1,38 @@ +/* generated using openapi-typescript-codegen -- do not edit */ +/* istanbul ignore file */ +/* tslint:disable */ +/* eslint-disable */ +import type { Account } from './Account'; +import type { Badge } from './Badge'; +import type { Configuration } from './Configuration'; +import type { GrantedAuthority } from './GrantedAuthority'; +import type { Point } from './Point'; +import type { Streak } from './Streak'; +export type User = { + id?: number; + firstName?: string; + lastName?: string; + email?: string; + checkingAccount?: Account; + savingsAccount?: Account; + password?: string; + createdAt?: string; + role?: User.role; + badges?: Array<Badge>; + point?: Point; + streak?: Streak; + configuration?: Configuration; + enabled?: boolean; + accountNonExpired?: boolean; + credentialsNonExpired?: boolean; + username?: string; + authorities?: Array<GrantedAuthority>; + accountNonLocked?: boolean; +}; +export namespace User { + export enum role { + USER = 'USER', + ADMIN = 'ADMIN', + } +} + diff --git a/src/api/services/ImageService.ts b/src/api/services/ImageService.ts new file mode 100644 index 0000000000000000000000000000000000000000..672da2fa200bba757b8ad192a39ad131e51aee3c --- /dev/null +++ b/src/api/services/ImageService.ts @@ -0,0 +1,48 @@ +/* generated using openapi-typescript-codegen -- do not edit */ +/* istanbul ignore file */ +/* tslint:disable */ +/* eslint-disable */ +import type { CancelablePromise } from '../core/CancelablePromise'; +import { OpenAPI } from '../core/OpenAPI'; +import { request as __request } from '../core/request'; +export class ImageService { + /** + * Upload an image + * Upload an image to the server + * @returns number Successfully uploaded the image + * @throws ApiError + */ + public static uploadImage({ + requestBody, + }: { + requestBody?: { + file: Blob; + }, + }): CancelablePromise<number> { + return __request(OpenAPI, { + method: 'POST', + url: '/api/image/upload', + body: requestBody, + mediaType: 'application/json', + }); + } + /** + * Retrieve an image + * Retrieve an image from the server + * @returns string Successfully retrieved the image + * @throws ApiError + */ + public static getImage({ + id, + }: { + id: number, + }): CancelablePromise<Array<string>> { + return __request(OpenAPI, { + method: 'GET', + url: '/api/image/{id}', + path: { + 'id': id, + }, + }); + } +} diff --git a/src/api/services/UserService.ts b/src/api/services/UserService.ts index d4bec02c95c05faf669d7d0663da47d34ff67771..32f920f9b91b3a9bfcabe9764f796d09747942e2 100644 --- a/src/api/services/UserService.ts +++ b/src/api/services/UserService.ts @@ -2,6 +2,13 @@ /* istanbul ignore file */ /* tslint:disable */ /* eslint-disable */ +import type { Account } from '../models/Account'; +import type { BankAccountDTO } from '../models/BankAccountDTO'; +import type { Budget } from '../models/Budget'; +import type { BudgetRequestDTO } from '../models/BudgetRequestDTO'; +import type { BudgetResponseDTO } from '../models/BudgetResponseDTO'; +import type { ExpenseRequestDTO } from '../models/ExpenseRequestDTO'; +import type { ExpenseResponseDTO } from '../models/ExpenseResponseDTO'; import type { PasswordResetDTO } from '../models/PasswordResetDTO'; import type { ProfileDTO } from '../models/ProfileDTO'; import type { UserDTO } from '../models/UserDTO'; @@ -49,6 +56,71 @@ export class UserService { }, }); } + /** + * Updates a budget + * Updates a budget based on the budget request + * @returns any Successfully updated budget + * @throws ApiError + */ + public static updateBudget({ + requestBody, + }: { + requestBody: BudgetResponseDTO, + }): CancelablePromise<Record<string, any>> { + return __request(OpenAPI, { + method: 'POST', + url: '/api/budget/update', + body: requestBody, + mediaType: 'application/json', + errors: { + 500: `Budget is not found`, + }, + }); + } + /** + * Created/Updates an expense + * Creates/Updates a budget based on the budget request + * @returns any Successfully updated budget + * @throws ApiError + */ + public static updateExpense({ + budgetId, + requestBody, + }: { + budgetId: number, + requestBody: ExpenseRequestDTO, + }): CancelablePromise<Record<string, any>> { + return __request(OpenAPI, { + method: 'POST', + url: '/api/budget/update/expense/{budgetId}', + path: { + 'budgetId': budgetId, + }, + body: requestBody, + mediaType: 'application/json', + errors: { + 500: `Error updating expense`, + }, + }); + } + /** + * Create a new budget + * Create a new budget with based on the budget request + * @returns Budget Successfully created new budget + * @throws ApiError + */ + public static createBudget({ + requestBody, + }: { + requestBody: BudgetRequestDTO, + }): CancelablePromise<Budget> { + return __request(OpenAPI, { + method: 'POST', + url: '/api/budget/create', + body: requestBody, + mediaType: 'application/json', + }); + } /** * Update a profile * Update the profile of the authenticated user @@ -67,6 +139,24 @@ export class UserService { mediaType: 'application/json', }); } + /** + * Update a user's bank account + * Changes either a user's checking account or savings account + * @returns Account OK + * @throws ApiError + */ + public static selectBankAccount({ + requestBody, + }: { + requestBody: BankAccountDTO, + }): CancelablePromise<Account> { + return __request(OpenAPI, { + method: 'PATCH', + url: '/api/users/update-account', + body: requestBody, + mediaType: 'application/json', + }); + } /** * Get a profile * Get the profile of a user @@ -98,4 +188,123 @@ export class UserService { url: '/api/users/me', }); } + /** + * Get the list of budgets + * Get all budgets related to the authenticated user + * @returns BudgetResponseDTO Successfully got budgets + * @throws ApiError + */ + public static getBudgetsByUser(): CancelablePromise<Array<BudgetResponseDTO>> { + return __request(OpenAPI, { + method: 'GET', + url: '/api/budget', + }); + } + /** + * Get the budget + * Get budget by its id + * @returns BudgetResponseDTO Successfully got budget + * @throws ApiError + */ + public static getBudget({ + budgetId, + }: { + budgetId: number, + }): CancelablePromise<BudgetResponseDTO> { + return __request(OpenAPI, { + method: 'GET', + url: '/api/budget/{budgetId}', + path: { + 'budgetId': budgetId, + }, + errors: { + 500: `Budget is not found`, + }, + }); + } + /** + * Get the list of budgets + * Get all budgets related to the authenticated user + * @returns ExpenseResponseDTO Successfully got expenses + * @throws ApiError + */ + public static getExpenses({ + budgetId, + }: { + budgetId: number, + }): CancelablePromise<Array<ExpenseResponseDTO>> { + return __request(OpenAPI, { + method: 'GET', + url: '/api/budget/expenses/{budgetId}', + path: { + 'budgetId': budgetId, + }, + }); + } + /** + * Get the expense + * Get expense by its id + * @returns ExpenseResponseDTO Successfully got expense + * @throws ApiError + */ + public static getExpense({ + expenseId, + }: { + expenseId: number, + }): CancelablePromise<ExpenseResponseDTO> { + return __request(OpenAPI, { + method: 'GET', + url: '/api/budget/expense/{expenseId}', + path: { + 'expenseId': expenseId, + }, + errors: { + 500: `Expense is not found`, + }, + }); + } + /** + * Deletes a budget + * Deletes a budget based on provided budget id + * @returns any Successfully deleted budget + * @throws ApiError + */ + public static deleteBudget({ + budgetId, + }: { + budgetId: number, + }): CancelablePromise<Record<string, any>> { + return __request(OpenAPI, { + method: 'GET', + url: '/api/budget/delete/{budgetId}', + path: { + 'budgetId': budgetId, + }, + errors: { + 500: `Budget is not found`, + }, + }); + } + /** + * Deletes an expense + * Deletes an expense based on provided expense id + * @returns any Successfully deleted expense + * @throws ApiError + */ + public static deleteExpense({ + expenseId, + }: { + expenseId: number, + }): CancelablePromise<Record<string, any>> { + return __request(OpenAPI, { + method: 'GET', + url: '/api/budget/delete/expense/{expenseId}', + path: { + 'expenseId': expenseId, + }, + errors: { + 500: `Expense is not found`, + }, + }); + } }