commit
1aa3ed85c5
35 changed files with 340 additions and 644 deletions
|
@ -5,4 +5,10 @@
|
||||||
#
|
#
|
||||||
|
|
||||||
# prisma database url
|
# prisma database url
|
||||||
DATABASE_URL="postgresql://prisma:prisma@localhost:5432/finances?schema=public"
|
DATABASE_URL='postgresql://prisma:prisma@localhost:5432/finances?schema=public'
|
||||||
|
|
||||||
|
AUTH0_SECRET=''
|
||||||
|
AUTH0_BASE_URL='http://localhost:3000'
|
||||||
|
AUTH0_ISSUER_BASE_URL=''
|
||||||
|
AUTH0_CLIENT_ID=''
|
||||||
|
AUTH0_CLIENT_SECRET=''
|
||||||
|
|
151
package-lock.json
generated
151
package-lock.json
generated
|
@ -8,6 +8,7 @@
|
||||||
"name": "next-finances",
|
"name": "next-finances",
|
||||||
"version": "1.2.0",
|
"version": "1.2.0",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
|
"@auth0/nextjs-auth0": "^3.5.0",
|
||||||
"@hookform/resolvers": "^3.3.4",
|
"@hookform/resolvers": "^3.3.4",
|
||||||
"@lucia-auth/adapter-prisma": "^4.0.0",
|
"@lucia-auth/adapter-prisma": "^4.0.0",
|
||||||
"@prisma/client": "^5.10.2",
|
"@prisma/client": "^5.10.2",
|
||||||
|
@ -79,6 +80,28 @@
|
||||||
"url": "https://github.com/sponsors/sindresorhus"
|
"url": "https://github.com/sponsors/sindresorhus"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/@auth0/nextjs-auth0": {
|
||||||
|
"version": "3.5.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/@auth0/nextjs-auth0/-/nextjs-auth0-3.5.0.tgz",
|
||||||
|
"integrity": "sha512-uFZEE2QQf1zU+jRK2fwqxRQt+WSqDPYF2tnr7d6BEa7b6L6tpPJ3evzoImbWSY1a7gFdvD7RD/Rvrsx7B5CKVg==",
|
||||||
|
"dependencies": {
|
||||||
|
"@panva/hkdf": "^1.0.2",
|
||||||
|
"cookie": "^0.6.0",
|
||||||
|
"debug": "^4.3.4",
|
||||||
|
"joi": "^17.6.0",
|
||||||
|
"jose": "^4.9.2",
|
||||||
|
"oauth4webapi": "^2.3.0",
|
||||||
|
"openid-client": "^5.2.1",
|
||||||
|
"tslib": "^2.4.0",
|
||||||
|
"url-join": "^4.0.1"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">=16"
|
||||||
|
},
|
||||||
|
"peerDependencies": {
|
||||||
|
"next": ">=10"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/@babel/runtime": {
|
"node_modules/@babel/runtime": {
|
||||||
"version": "7.24.0",
|
"version": "7.24.0",
|
||||||
"resolved": "https://registry.npmjs.org/@babel/runtime/-/runtime-7.24.0.tgz",
|
"resolved": "https://registry.npmjs.org/@babel/runtime/-/runtime-7.24.0.tgz",
|
||||||
|
@ -202,6 +225,19 @@
|
||||||
"resolved": "https://registry.npmjs.org/@floating-ui/utils/-/utils-0.2.1.tgz",
|
"resolved": "https://registry.npmjs.org/@floating-ui/utils/-/utils-0.2.1.tgz",
|
||||||
"integrity": "sha512-9TANp6GPoMtYzQdt54kfAyMmz1+osLlXdg2ENroU7zzrtflTLrrC/lgrIfaSe+Wu0b89GKccT7vxXA0MoAIO+Q=="
|
"integrity": "sha512-9TANp6GPoMtYzQdt54kfAyMmz1+osLlXdg2ENroU7zzrtflTLrrC/lgrIfaSe+Wu0b89GKccT7vxXA0MoAIO+Q=="
|
||||||
},
|
},
|
||||||
|
"node_modules/@hapi/hoek": {
|
||||||
|
"version": "9.3.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/@hapi/hoek/-/hoek-9.3.0.tgz",
|
||||||
|
"integrity": "sha512-/c6rf4UJlmHlC9b5BaNvzAcFv7HZ2QHaV0D4/HNlBdvFnvQq8RI4kYdhyPCl7Xj+oWvTWQ8ujhqS53LIgAe6KQ=="
|
||||||
|
},
|
||||||
|
"node_modules/@hapi/topo": {
|
||||||
|
"version": "5.1.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/@hapi/topo/-/topo-5.1.0.tgz",
|
||||||
|
"integrity": "sha512-foQZKJig7Ob0BMAYBfcJk8d77QtOe7Wo4ox7ff1lQYoNNAb6jwcY1ncdoy2e9wQZzvNy7ODZCYJkK8kzmcAnAg==",
|
||||||
|
"dependencies": {
|
||||||
|
"@hapi/hoek": "^9.0.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/@hookform/resolvers": {
|
"node_modules/@hookform/resolvers": {
|
||||||
"version": "3.3.4",
|
"version": "3.3.4",
|
||||||
"resolved": "https://registry.npmjs.org/@hookform/resolvers/-/resolvers-3.3.4.tgz",
|
"resolved": "https://registry.npmjs.org/@hookform/resolvers/-/resolvers-3.3.4.tgz",
|
||||||
|
@ -609,6 +645,14 @@
|
||||||
"node": ">= 8"
|
"node": ">= 8"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/@panva/hkdf": {
|
||||||
|
"version": "1.1.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/@panva/hkdf/-/hkdf-1.1.1.tgz",
|
||||||
|
"integrity": "sha512-dhPeilub1NuIG0X5Kvhh9lH4iW3ZsHlnzwgwbOlgwQ2wG1IqFzsgHqmKPk3WzsdWAeaxKJxgM0+W433RmN45GA==",
|
||||||
|
"funding": {
|
||||||
|
"url": "https://github.com/sponsors/panva"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/@pkgjs/parseargs": {
|
"node_modules/@pkgjs/parseargs": {
|
||||||
"version": "0.11.0",
|
"version": "0.11.0",
|
||||||
"resolved": "https://registry.npmjs.org/@pkgjs/parseargs/-/parseargs-0.11.0.tgz",
|
"resolved": "https://registry.npmjs.org/@pkgjs/parseargs/-/parseargs-0.11.0.tgz",
|
||||||
|
@ -1890,6 +1934,24 @@
|
||||||
"@types/trusted-types": "2.0.7"
|
"@types/trusted-types": "2.0.7"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/@sideway/address": {
|
||||||
|
"version": "4.1.5",
|
||||||
|
"resolved": "https://registry.npmjs.org/@sideway/address/-/address-4.1.5.tgz",
|
||||||
|
"integrity": "sha512-IqO/DUQHUkPeixNQ8n0JA6102hT9CmaljNTPmQ1u8MEhBo/R4Q8eKLN/vGZxuebwOroDB4cbpjheD4+/sKFK4Q==",
|
||||||
|
"dependencies": {
|
||||||
|
"@hapi/hoek": "^9.0.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/@sideway/formula": {
|
||||||
|
"version": "3.0.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/@sideway/formula/-/formula-3.0.1.tgz",
|
||||||
|
"integrity": "sha512-/poHZJJVjx3L+zVD6g9KgHfYnb443oi7wLu/XKojDviHy6HOEOA6z1Trk5aR1dGcmPenJEgb2sK2I80LeS3MIg=="
|
||||||
|
},
|
||||||
|
"node_modules/@sideway/pinpoint": {
|
||||||
|
"version": "2.0.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/@sideway/pinpoint/-/pinpoint-2.0.0.tgz",
|
||||||
|
"integrity": "sha512-RNiOoTPkptFtSVzQevY/yWtZwf/RxyVnPy/OcA9HBM3MlGDnBEYL5B41H0MTn0Uec8Hi+2qUtTfG2WWZBmMejQ=="
|
||||||
|
},
|
||||||
"node_modules/@swc/helpers": {
|
"node_modules/@swc/helpers": {
|
||||||
"version": "0.5.2",
|
"version": "0.5.2",
|
||||||
"resolved": "https://registry.npmjs.org/@swc/helpers/-/helpers-0.5.2.tgz",
|
"resolved": "https://registry.npmjs.org/@swc/helpers/-/helpers-0.5.2.tgz",
|
||||||
|
@ -3041,6 +3103,14 @@
|
||||||
"resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz",
|
"resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz",
|
||||||
"integrity": "sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg=="
|
"integrity": "sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg=="
|
||||||
},
|
},
|
||||||
|
"node_modules/cookie": {
|
||||||
|
"version": "0.6.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/cookie/-/cookie-0.6.0.tgz",
|
||||||
|
"integrity": "sha512-U71cyTamuh1CRNCfpGY6to28lxvNwPG4Guz/EVjgf3Jmzv0vlDp1atT9eS5dDjMYHucpHbWns6Lwf3BKz6svdw==",
|
||||||
|
"engines": {
|
||||||
|
"node": ">= 0.6"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/create-require": {
|
"node_modules/create-require": {
|
||||||
"version": "1.1.1",
|
"version": "1.1.1",
|
||||||
"resolved": "https://registry.npmjs.org/create-require/-/create-require-1.1.1.tgz",
|
"resolved": "https://registry.npmjs.org/create-require/-/create-require-1.1.1.tgz",
|
||||||
|
@ -3096,7 +3166,6 @@
|
||||||
"version": "4.3.4",
|
"version": "4.3.4",
|
||||||
"resolved": "https://registry.npmjs.org/debug/-/debug-4.3.4.tgz",
|
"resolved": "https://registry.npmjs.org/debug/-/debug-4.3.4.tgz",
|
||||||
"integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==",
|
"integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==",
|
||||||
"dev": true,
|
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"ms": "2.1.2"
|
"ms": "2.1.2"
|
||||||
},
|
},
|
||||||
|
@ -4912,6 +4981,26 @@
|
||||||
"jiti": "bin/jiti.js"
|
"jiti": "bin/jiti.js"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/joi": {
|
||||||
|
"version": "17.12.3",
|
||||||
|
"resolved": "https://registry.npmjs.org/joi/-/joi-17.12.3.tgz",
|
||||||
|
"integrity": "sha512-2RRziagf555owrm9IRVtdKynOBeITiDpuZqIpgwqXShPncPKNiRQoiGsl/T8SQdq+8ugRzH2LqY67irr2y/d+g==",
|
||||||
|
"dependencies": {
|
||||||
|
"@hapi/hoek": "^9.3.0",
|
||||||
|
"@hapi/topo": "^5.1.0",
|
||||||
|
"@sideway/address": "^4.1.5",
|
||||||
|
"@sideway/formula": "^3.0.1",
|
||||||
|
"@sideway/pinpoint": "^2.0.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/jose": {
|
||||||
|
"version": "4.15.5",
|
||||||
|
"resolved": "https://registry.npmjs.org/jose/-/jose-4.15.5.tgz",
|
||||||
|
"integrity": "sha512-jc7BFxgKPKi94uOvEmzlSWFFe2+vASyXaKUpdQKatWAESU2MWjDfFf0fdfc83CDKcA5QecabZeNLyfhe3yKNkg==",
|
||||||
|
"funding": {
|
||||||
|
"url": "https://github.com/sponsors/panva"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/js-tokens": {
|
"node_modules/js-tokens": {
|
||||||
"version": "4.0.0",
|
"version": "4.0.0",
|
||||||
"resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz",
|
"resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz",
|
||||||
|
@ -5309,8 +5398,7 @@
|
||||||
"node_modules/ms": {
|
"node_modules/ms": {
|
||||||
"version": "2.1.2",
|
"version": "2.1.2",
|
||||||
"resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz",
|
"resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz",
|
||||||
"integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==",
|
"integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w=="
|
||||||
"dev": true
|
|
||||||
},
|
},
|
||||||
"node_modules/mz": {
|
"node_modules/mz": {
|
||||||
"version": "2.7.0",
|
"version": "2.7.0",
|
||||||
|
@ -5455,6 +5543,14 @@
|
||||||
"node": ">=0.10.0"
|
"node": ">=0.10.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/oauth4webapi": {
|
||||||
|
"version": "2.10.4",
|
||||||
|
"resolved": "https://registry.npmjs.org/oauth4webapi/-/oauth4webapi-2.10.4.tgz",
|
||||||
|
"integrity": "sha512-DSoj8QoChzOCQlJkRmYxAJCIpnXFW32R0Uq7avyghIeB6iJq0XAblOD7pcq3mx4WEBDwMuKr0Y1qveCBleG2Xw==",
|
||||||
|
"funding": {
|
||||||
|
"url": "https://github.com/sponsors/panva"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/object-assign": {
|
"node_modules/object-assign": {
|
||||||
"version": "4.1.1",
|
"version": "4.1.1",
|
||||||
"resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz",
|
"resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz",
|
||||||
|
@ -5581,6 +5677,14 @@
|
||||||
"url": "https://github.com/sponsors/ljharb"
|
"url": "https://github.com/sponsors/ljharb"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/oidc-token-hash": {
|
||||||
|
"version": "5.0.3",
|
||||||
|
"resolved": "https://registry.npmjs.org/oidc-token-hash/-/oidc-token-hash-5.0.3.tgz",
|
||||||
|
"integrity": "sha512-IF4PcGgzAr6XXSff26Sk/+P4KZFJVuHAJZj3wgO3vX2bMdNVp/QXTP3P7CEm9V1IdG8lDLY3HhiqpsE/nOwpPw==",
|
||||||
|
"engines": {
|
||||||
|
"node": "^10.13.0 || >=12.0.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/once": {
|
"node_modules/once": {
|
||||||
"version": "1.4.0",
|
"version": "1.4.0",
|
||||||
"resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz",
|
"resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz",
|
||||||
|
@ -5589,6 +5693,39 @@
|
||||||
"wrappy": "1"
|
"wrappy": "1"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/openid-client": {
|
||||||
|
"version": "5.6.5",
|
||||||
|
"resolved": "https://registry.npmjs.org/openid-client/-/openid-client-5.6.5.tgz",
|
||||||
|
"integrity": "sha512-5P4qO9nGJzB5PI0LFlhj4Dzg3m4odt0qsJTfyEtZyOlkgpILwEioOhVVJOrS1iVH494S4Ee5OCjjg6Bf5WOj3w==",
|
||||||
|
"dependencies": {
|
||||||
|
"jose": "^4.15.5",
|
||||||
|
"lru-cache": "^6.0.0",
|
||||||
|
"object-hash": "^2.2.0",
|
||||||
|
"oidc-token-hash": "^5.0.3"
|
||||||
|
},
|
||||||
|
"funding": {
|
||||||
|
"url": "https://github.com/sponsors/panva"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/openid-client/node_modules/lru-cache": {
|
||||||
|
"version": "6.0.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-6.0.0.tgz",
|
||||||
|
"integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==",
|
||||||
|
"dependencies": {
|
||||||
|
"yallist": "^4.0.0"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">=10"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/openid-client/node_modules/object-hash": {
|
||||||
|
"version": "2.2.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/object-hash/-/object-hash-2.2.0.tgz",
|
||||||
|
"integrity": "sha512-gScRMn0bS5fH+IuwyIFgnh9zBdo4DV+6GhygmWM9HyNJSgS0hScp1f5vjtm7oIIOiT9trXrShAkLFSc2IqKNgw==",
|
||||||
|
"engines": {
|
||||||
|
"node": ">= 6"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/optionator": {
|
"node_modules/optionator": {
|
||||||
"version": "0.9.3",
|
"version": "0.9.3",
|
||||||
"resolved": "https://registry.npmjs.org/optionator/-/optionator-0.9.3.tgz",
|
"resolved": "https://registry.npmjs.org/optionator/-/optionator-0.9.3.tgz",
|
||||||
|
@ -7261,6 +7398,11 @@
|
||||||
"punycode": "^2.1.0"
|
"punycode": "^2.1.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/url-join": {
|
||||||
|
"version": "4.0.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/url-join/-/url-join-4.0.1.tgz",
|
||||||
|
"integrity": "sha512-jk1+QP6ZJqyOiuEI9AEWQfju/nB2Pw466kbA0LEZljHwKeMgd9WrAEgEGxjPDD2+TNbbb37rTyhEfrCXfuKXnA=="
|
||||||
|
},
|
||||||
"node_modules/use-callback-ref": {
|
"node_modules/use-callback-ref": {
|
||||||
"version": "1.3.1",
|
"version": "1.3.1",
|
||||||
"resolved": "https://registry.npmjs.org/use-callback-ref/-/use-callback-ref-1.3.1.tgz",
|
"resolved": "https://registry.npmjs.org/use-callback-ref/-/use-callback-ref-1.3.1.tgz",
|
||||||
|
@ -7624,8 +7766,7 @@
|
||||||
"node_modules/yallist": {
|
"node_modules/yallist": {
|
||||||
"version": "4.0.0",
|
"version": "4.0.0",
|
||||||
"resolved": "https://registry.npmjs.org/yallist/-/yallist-4.0.0.tgz",
|
"resolved": "https://registry.npmjs.org/yallist/-/yallist-4.0.0.tgz",
|
||||||
"integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==",
|
"integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A=="
|
||||||
"dev": true
|
|
||||||
},
|
},
|
||||||
"node_modules/yaml": {
|
"node_modules/yaml": {
|
||||||
"version": "2.4.1",
|
"version": "2.4.1",
|
||||||
|
|
|
@ -18,6 +18,7 @@
|
||||||
"lint": "next lint"
|
"lint": "next lint"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
|
"@auth0/nextjs-auth0": "^3.5.0",
|
||||||
"@hookform/resolvers": "^3.3.4",
|
"@hookform/resolvers": "^3.3.4",
|
||||||
"@lucia-auth/adapter-prisma": "^4.0.0",
|
"@lucia-auth/adapter-prisma": "^4.0.0",
|
||||||
"@prisma/client": "^5.10.2",
|
"@prisma/client": "^5.10.2",
|
||||||
|
|
|
@ -0,0 +1,28 @@
|
||||||
|
/*
|
||||||
|
Warnings:
|
||||||
|
|
||||||
|
- You are about to drop the `lucia_session` table. If the table is not empty, all the data it contains will be lost.
|
||||||
|
- You are about to drop the `lucia_user` table. If the table is not empty, all the data it contains will be lost.
|
||||||
|
|
||||||
|
*/
|
||||||
|
-- DropForeignKey
|
||||||
|
ALTER TABLE "categories"
|
||||||
|
DROP CONSTRAINT "categories_user_id_fkey";
|
||||||
|
|
||||||
|
-- DropForeignKey
|
||||||
|
ALTER TABLE "entities"
|
||||||
|
DROP CONSTRAINT "entities_user_id_fkey";
|
||||||
|
|
||||||
|
-- DropForeignKey
|
||||||
|
ALTER TABLE "lucia_session"
|
||||||
|
DROP CONSTRAINT "lucia_session_userId_fkey";
|
||||||
|
|
||||||
|
-- DropForeignKey
|
||||||
|
ALTER TABLE "payments"
|
||||||
|
DROP CONSTRAINT "payments_user_id_fkey";
|
||||||
|
|
||||||
|
-- DropTable
|
||||||
|
DROP TABLE "lucia_session";
|
||||||
|
|
||||||
|
-- DropTable
|
||||||
|
DROP TABLE "lucia_user";
|
|
@ -7,36 +7,9 @@ datasource db {
|
||||||
url = env("DATABASE_URL")
|
url = env("DATABASE_URL")
|
||||||
}
|
}
|
||||||
|
|
||||||
model User {
|
|
||||||
// lucia internal fields
|
|
||||||
id String @id
|
|
||||||
sessions Session[]
|
|
||||||
|
|
||||||
// custom fields
|
|
||||||
username String @unique
|
|
||||||
password String
|
|
||||||
|
|
||||||
entities Entity[]
|
|
||||||
payments Payment[]
|
|
||||||
categories Category[]
|
|
||||||
|
|
||||||
@@map("lucia_user")
|
|
||||||
}
|
|
||||||
|
|
||||||
model Session {
|
|
||||||
// lucia internal fields
|
|
||||||
id String @id
|
|
||||||
userId String
|
|
||||||
expiresAt DateTime
|
|
||||||
user User @relation(references: [id], fields: [userId], onDelete: Cascade)
|
|
||||||
|
|
||||||
@@map("lucia_session")
|
|
||||||
}
|
|
||||||
|
|
||||||
model Entity {
|
model Entity {
|
||||||
id Int @id @default(autoincrement())
|
id Int @id @default(autoincrement())
|
||||||
userId String @map("user_id")
|
userId String @map("user_id")
|
||||||
user User @relation(fields: [userId], references: [id])
|
|
||||||
name String
|
name String
|
||||||
type EntityType
|
type EntityType
|
||||||
defaultCategory Category? @relation(fields: [defaultCategoryId], references: [id])
|
defaultCategory Category? @relation(fields: [defaultCategoryId], references: [id])
|
||||||
|
@ -59,7 +32,6 @@ enum EntityType {
|
||||||
model Payment {
|
model Payment {
|
||||||
id Int @id @default(autoincrement())
|
id Int @id @default(autoincrement())
|
||||||
userId String @map("user_id")
|
userId String @map("user_id")
|
||||||
user User @relation(fields: [userId], references: [id])
|
|
||||||
amount Int
|
amount Int
|
||||||
currency String @default("EUR")
|
currency String @default("EUR")
|
||||||
date DateTime @default(now())
|
date DateTime @default(now())
|
||||||
|
@ -79,7 +51,6 @@ model Payment {
|
||||||
model Category {
|
model Category {
|
||||||
id Int @id @default(autoincrement())
|
id Int @id @default(autoincrement())
|
||||||
userId String @map("user_id")
|
userId String @map("user_id")
|
||||||
user User @relation(fields: [userId], references: [id])
|
|
||||||
name String
|
name String
|
||||||
color String
|
color String
|
||||||
createdAt DateTime @default(now()) @map("created_at")
|
createdAt DateTime @default(now()) @map("created_at")
|
||||||
|
|
|
@ -1,42 +1,37 @@
|
||||||
import { Card, CardContent, CardDescription, CardFooter, CardHeader, CardTitle } from '@/components/ui/card';
|
import { Card, CardContent, CardDescription, CardFooter, CardHeader, CardTitle } from '@/components/ui/card';
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { getUser } from '@/auth';
|
|
||||||
import { redirect } from 'next/navigation';
|
|
||||||
import signOut from '@/lib/actions/signOut';
|
|
||||||
import { Label } from '@/components/ui/label';
|
import { Label } from '@/components/ui/label';
|
||||||
import { Input } from '@/components/ui/input';
|
import { Input } from '@/components/ui/input';
|
||||||
import { URL_SIGN_IN } from '@/lib/constants';
|
|
||||||
import generateSampleData from '@/lib/actions/generateSampleData';
|
import generateSampleData from '@/lib/actions/generateSampleData';
|
||||||
import prisma from '@/prisma';
|
import prisma from '@/prisma';
|
||||||
import { ServerActionTrigger } from '@/components/form/serverActionTrigger';
|
import { ServerActionTrigger } from '@/components/form/serverActionTrigger';
|
||||||
import accountDelete from '@/lib/actions/accountDelete';
|
import clearAccountData from '@/lib/actions/clearAccountData';
|
||||||
|
import { Button } from '@/components/ui/button';
|
||||||
|
import { getSession, Session } from '@auth0/nextjs-auth0';
|
||||||
|
import { URL_SIGN_OUT } from '@/lib/constants';
|
||||||
|
|
||||||
export default async function AccountPage() {
|
export default async function AccountPage() {
|
||||||
|
|
||||||
const user = await getUser();
|
const {user} = await getSession() as Session;
|
||||||
|
|
||||||
if (!user) {
|
|
||||||
redirect(URL_SIGN_IN);
|
|
||||||
}
|
|
||||||
|
|
||||||
let paymentCount = 0;
|
let paymentCount = 0;
|
||||||
paymentCount = await prisma.payment.count({
|
paymentCount = await prisma.payment.count({
|
||||||
where: {
|
where: {
|
||||||
userId: user.id,
|
userId: user.sub,
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
let entityCount = 0;
|
let entityCount = 0;
|
||||||
entityCount = await prisma.entity.count({
|
entityCount = await prisma.entity.count({
|
||||||
where: {
|
where: {
|
||||||
userId: user.id,
|
userId: user.sub,
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
let categoryCount = 0;
|
let categoryCount = 0;
|
||||||
categoryCount = await prisma.category.count({
|
categoryCount = await prisma.category.count({
|
||||||
where: {
|
where: {
|
||||||
userId: user.id,
|
userId: user.sub,
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -44,7 +39,7 @@ export default async function AccountPage() {
|
||||||
<div className="flex flex-col items-center">
|
<div className="flex flex-col items-center">
|
||||||
<Card className="w-full max-w-md md:mt-12">
|
<Card className="w-full max-w-md md:mt-12">
|
||||||
<CardHeader>
|
<CardHeader>
|
||||||
<CardTitle>Hey, {user?.username}!</CardTitle>
|
<CardTitle>Hey, {user.name}!</CardTitle>
|
||||||
<CardDescription>This is your account overview.</CardDescription>
|
<CardDescription>This is your account overview.</CardDescription>
|
||||||
</CardHeader>
|
</CardHeader>
|
||||||
<CardContent className="space-y-2">
|
<CardContent className="space-y-2">
|
||||||
|
@ -52,13 +47,13 @@ export default async function AccountPage() {
|
||||||
<Label>ID</Label>
|
<Label>ID</Label>
|
||||||
<Input
|
<Input
|
||||||
disabled
|
disabled
|
||||||
value={user?.id}/>
|
value={user.sub}/>
|
||||||
</div>
|
</div>
|
||||||
<div>
|
<div>
|
||||||
<Label>Username</Label>
|
<Label>Username</Label>
|
||||||
<Input
|
<Input
|
||||||
disabled
|
disabled
|
||||||
value={user?.username}/>
|
value={user.name}/>
|
||||||
</div>
|
</div>
|
||||||
<div className="flex flex-row items-center space-x-4">
|
<div className="flex flex-row items-center space-x-4">
|
||||||
<div>
|
<div>
|
||||||
|
@ -83,19 +78,21 @@ export default async function AccountPage() {
|
||||||
</CardContent>
|
</CardContent>
|
||||||
<CardFooter className="w-full grid gap-4 grid-cols-1 md:grid-cols-2">
|
<CardFooter className="w-full grid gap-4 grid-cols-1 md:grid-cols-2">
|
||||||
<ServerActionTrigger
|
<ServerActionTrigger
|
||||||
action={accountDelete}
|
action={clearAccountData}
|
||||||
dialog={{
|
dialog={{
|
||||||
title: 'Delete Account',
|
title: 'Clear account data',
|
||||||
description: 'Are you sure you want to delete your account? This action is irreversible.',
|
description: 'Are you sure you want to delete all payments, entities and categories from you account? This action is irreversible.',
|
||||||
actionText: 'Delete Account',
|
actionText: 'Clear data',
|
||||||
|
actionVariant: 'destructive',
|
||||||
}}
|
}}
|
||||||
variant="outline">
|
variant="outline">
|
||||||
Delete Account
|
Clear data
|
||||||
</ServerActionTrigger>
|
|
||||||
<ServerActionTrigger
|
|
||||||
action={signOut}>
|
|
||||||
Sign Out
|
|
||||||
</ServerActionTrigger>
|
</ServerActionTrigger>
|
||||||
|
<a href={URL_SIGN_OUT}>
|
||||||
|
<Button className="w-full">
|
||||||
|
Sign Out
|
||||||
|
</Button>
|
||||||
|
</a>
|
||||||
{
|
{
|
||||||
process.env.NODE_ENV === 'development' && (
|
process.env.NODE_ENV === 'development' && (
|
||||||
<ServerActionTrigger
|
<ServerActionTrigger
|
||||||
|
|
3
src/app/api/auth/[auth0]/route.js
Normal file
3
src/app/api/auth/[auth0]/route.js
Normal file
|
@ -0,0 +1,3 @@
|
||||||
|
import { handleAuth } from '@auth0/nextjs-auth0';
|
||||||
|
|
||||||
|
export const GET = handleAuth();
|
|
@ -1,13 +0,0 @@
|
||||||
import React from 'react';
|
|
||||||
|
|
||||||
export default function AuthLayout({
|
|
||||||
children,
|
|
||||||
}: Readonly<{
|
|
||||||
children: React.ReactNode;
|
|
||||||
}>) {
|
|
||||||
return (
|
|
||||||
<div className="flex justify-center">
|
|
||||||
{children}
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
}
|
|
|
@ -1,25 +0,0 @@
|
||||||
import React from 'react';
|
|
||||||
import { Card, CardContent, CardDescription, CardFooter, CardHeader, CardTitle } from '@/components/ui/card';
|
|
||||||
import SignInForm from '@/components/form/signInForm';
|
|
||||||
import signIn from '@/lib/actions/signIn';
|
|
||||||
import Link from 'next/link';
|
|
||||||
import { URL_SIGN_UP } from '@/lib/constants';
|
|
||||||
|
|
||||||
export default async function SignInPage() {
|
|
||||||
return (
|
|
||||||
<Card className="w-full max-w-md mt-12">
|
|
||||||
<CardHeader>
|
|
||||||
<CardTitle>Sign in</CardTitle>
|
|
||||||
<CardDescription>Sign into your existing account</CardDescription>
|
|
||||||
</CardHeader>
|
|
||||||
<CardContent>
|
|
||||||
<SignInForm onSubmit={signIn}/>
|
|
||||||
</CardContent>
|
|
||||||
<CardFooter>
|
|
||||||
<Link href={URL_SIGN_UP}>
|
|
||||||
Don't have an account? Sign up
|
|
||||||
</Link>
|
|
||||||
</CardFooter>
|
|
||||||
</Card>
|
|
||||||
);
|
|
||||||
}
|
|
|
@ -1,25 +0,0 @@
|
||||||
import React from 'react';
|
|
||||||
import { Card, CardContent, CardDescription, CardFooter, CardHeader, CardTitle } from '@/components/ui/card';
|
|
||||||
import signUp from '@/lib/actions/signUp';
|
|
||||||
import SignUpForm from '@/components/form/signUpForm';
|
|
||||||
import Link from 'next/link';
|
|
||||||
import { URL_SIGN_IN } from '@/lib/constants';
|
|
||||||
|
|
||||||
export default async function SignUpPage() {
|
|
||||||
return (
|
|
||||||
<Card className="w-full max-w-md mt-12">
|
|
||||||
<CardHeader>
|
|
||||||
<CardTitle>Sign up</CardTitle>
|
|
||||||
<CardDescription>Create a new account.</CardDescription>
|
|
||||||
</CardHeader>
|
|
||||||
<CardContent>
|
|
||||||
<SignUpForm onSubmit={signUp}/>
|
|
||||||
</CardContent>
|
|
||||||
<CardFooter>
|
|
||||||
<Link href={URL_SIGN_IN}>
|
|
||||||
Already have an account? Sign in
|
|
||||||
</Link>
|
|
||||||
</CardFooter>
|
|
||||||
</Card>
|
|
||||||
);
|
|
||||||
}
|
|
|
@ -1,13 +1,13 @@
|
||||||
import { getUser } from '@/auth';
|
|
||||||
import prisma from '@/prisma';
|
import prisma from '@/prisma';
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
import CategoryPageClientContent from '@/components/categoryPageClientComponents';
|
import CategoryPageClientContent from '@/components/categoryPageClientComponents';
|
||||||
import categoryCreateUpdate from '@/lib/actions/categoryCreateUpdate';
|
import categoryCreateUpdate from '@/lib/actions/categoryCreateUpdate';
|
||||||
import categoryDelete from '@/lib/actions/categoryDelete';
|
import categoryDelete from '@/lib/actions/categoryDelete';
|
||||||
|
import { getSession, Session } from '@auth0/nextjs-auth0';
|
||||||
|
|
||||||
export default async function CategoriesPage() {
|
export default async function CategoriesPage() {
|
||||||
|
|
||||||
const user = await getUser();
|
const {user} = await getSession() as Session;
|
||||||
|
|
||||||
const categories = await prisma.category.findMany({
|
const categories = await prisma.category.findMany({
|
||||||
where: {
|
where: {
|
||||||
|
|
|
@ -1,13 +1,13 @@
|
||||||
import prisma from '@/prisma';
|
import prisma from '@/prisma';
|
||||||
import { getUser } from '@/auth';
|
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
import EntityPageClientContent from '@/components/entityPageClientComponents';
|
import EntityPageClientContent from '@/components/entityPageClientComponents';
|
||||||
import entityCreateUpdate from '@/lib/actions/entityCreateUpdate';
|
import entityCreateUpdate from '@/lib/actions/entityCreateUpdate';
|
||||||
import entityDelete from '@/lib/actions/entityDelete';
|
import entityDelete from '@/lib/actions/entityDelete';
|
||||||
|
import { getSession, Session } from '@auth0/nextjs-auth0';
|
||||||
|
|
||||||
export default async function EntitiesPage() {
|
export default async function EntitiesPage() {
|
||||||
|
|
||||||
const user = await getUser();
|
const {user} = await getSession() as Session;
|
||||||
|
|
||||||
const entities = await prisma.entity.findMany({
|
const entities = await prisma.entity.findMany({
|
||||||
where: {
|
where: {
|
||||||
|
|
|
@ -5,6 +5,7 @@ import { cn } from '@/lib/utils';
|
||||||
import { Toaster } from '@/components/ui/sonner';
|
import { Toaster } from '@/components/ui/sonner';
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
import Navigation from '@/components/navigation';
|
import Navigation from '@/components/navigation';
|
||||||
|
import { UserProvider } from '@auth0/nextjs-auth0/client';
|
||||||
|
|
||||||
const inter = Inter({subsets: ['latin']});
|
const inter = Inter({subsets: ['latin']});
|
||||||
|
|
||||||
|
@ -49,6 +50,7 @@ export default function RootLayout({
|
||||||
href="/logo_white.png"
|
href="/logo_white.png"
|
||||||
/>
|
/>
|
||||||
</head>
|
</head>
|
||||||
|
<UserProvider>
|
||||||
<body className={cn('dark', inter.className)}>
|
<body className={cn('dark', inter.className)}>
|
||||||
<Navigation/>
|
<Navigation/>
|
||||||
<main className="p-4 sm:p-8">
|
<main className="p-4 sm:p-8">
|
||||||
|
@ -56,6 +58,7 @@ export default function RootLayout({
|
||||||
</main>
|
</main>
|
||||||
<Toaster/>
|
<Toaster/>
|
||||||
</body>
|
</body>
|
||||||
|
</UserProvider>
|
||||||
</html>
|
</html>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
|
@ -2,8 +2,8 @@ import React from 'react';
|
||||||
import { Category, Entity, EntityType } from '@prisma/client';
|
import { Category, Entity, EntityType } from '@prisma/client';
|
||||||
import { Scope, ScopeType } from '@/lib/types/scope';
|
import { Scope, ScopeType } from '@/lib/types/scope';
|
||||||
import prisma from '@/prisma';
|
import prisma from '@/prisma';
|
||||||
import { getUser } from '@/auth';
|
|
||||||
import DashboardPageClient from '@/components/dashboardPageClientComponents';
|
import DashboardPageClient from '@/components/dashboardPageClientComponents';
|
||||||
|
import { getSession, Session } from '@auth0/nextjs-auth0';
|
||||||
|
|
||||||
export type CategoryNumber = {
|
export type CategoryNumber = {
|
||||||
category: Category,
|
category: Category,
|
||||||
|
@ -17,17 +17,14 @@ export type EntityNumber = {
|
||||||
|
|
||||||
export default async function DashboardPage(props: { searchParams?: { scope: ScopeType } }) {
|
export default async function DashboardPage(props: { searchParams?: { scope: ScopeType } }) {
|
||||||
|
|
||||||
const user = await getUser();
|
const {user} = await getSession() as Session;
|
||||||
if (!user) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const scope = Scope.of(props.searchParams?.scope || ScopeType.ThisMonth);
|
const scope = Scope.of(props.searchParams?.scope || ScopeType.ThisMonth);
|
||||||
|
|
||||||
// get all payments in the current scope
|
// get all payments in the current scope
|
||||||
const payments = await prisma.payment.findMany({
|
const payments = await prisma.payment.findMany({
|
||||||
where: {
|
where: {
|
||||||
userId: user?.id,
|
userId: user.sub,
|
||||||
date: {
|
date: {
|
||||||
gte: scope.start,
|
gte: scope.start,
|
||||||
lte: scope.end,
|
lte: scope.end,
|
||||||
|
|
|
@ -1,17 +1,17 @@
|
||||||
import { getUser } from '@/auth';
|
|
||||||
import prisma from '@/prisma';
|
import prisma from '@/prisma';
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
import PaymentPageClientContent from '@/components/paymentPageClientComponents';
|
import PaymentPageClientContent from '@/components/paymentPageClientComponents';
|
||||||
import paymentCreateUpdate from '@/lib/actions/paymentCreateUpdate';
|
import paymentCreateUpdate from '@/lib/actions/paymentCreateUpdate';
|
||||||
import paymentDelete from '@/lib/actions/paymentDelete';
|
import paymentDelete from '@/lib/actions/paymentDelete';
|
||||||
|
import { getSession, Session } from '@auth0/nextjs-auth0';
|
||||||
|
|
||||||
export default async function PaymentsPage() {
|
export default async function PaymentsPage() {
|
||||||
|
|
||||||
const user = await getUser();
|
const {user} = await getSession() as Session;
|
||||||
|
|
||||||
const payments = await prisma.payment.findMany({
|
const payments = await prisma.payment.findMany({
|
||||||
where: {
|
where: {
|
||||||
userId: user?.id,
|
userId: user.sub,
|
||||||
},
|
},
|
||||||
orderBy: [
|
orderBy: [
|
||||||
{
|
{
|
||||||
|
@ -25,7 +25,7 @@ export default async function PaymentsPage() {
|
||||||
|
|
||||||
const entities = await prisma.entity.findMany({
|
const entities = await prisma.entity.findMany({
|
||||||
where: {
|
where: {
|
||||||
userId: user?.id,
|
userId: user.sub,
|
||||||
},
|
},
|
||||||
orderBy: [
|
orderBy: [
|
||||||
{
|
{
|
||||||
|
@ -39,7 +39,7 @@ export default async function PaymentsPage() {
|
||||||
|
|
||||||
const categories = await prisma.category.findMany({
|
const categories = await prisma.category.findMany({
|
||||||
where: {
|
where: {
|
||||||
userId: user?.id,
|
userId: user.sub,
|
||||||
},
|
},
|
||||||
orderBy: [
|
orderBy: [
|
||||||
{
|
{
|
||||||
|
|
67
src/auth.ts
67
src/auth.ts
|
@ -1,67 +0,0 @@
|
||||||
import { Lucia } from 'lucia';
|
|
||||||
import { PrismaAdapter } from '@lucia-auth/adapter-prisma';
|
|
||||||
import { cookies } from 'next/headers';
|
|
||||||
import prisma from '@/prisma';
|
|
||||||
|
|
||||||
const adapter = new PrismaAdapter(prisma.session, prisma.user);
|
|
||||||
|
|
||||||
export const lucia = new Lucia(adapter, {
|
|
||||||
sessionCookie: {
|
|
||||||
expires: false,
|
|
||||||
attributes: {
|
|
||||||
sameSite: 'strict',
|
|
||||||
domain: process.env.NODE_ENV === 'production' ? process.env.COOKIE_DOMAIN : undefined,
|
|
||||||
secure: process.env.NODE_ENV === 'production',
|
|
||||||
},
|
|
||||||
},
|
|
||||||
getUserAttributes: (attributes) => {
|
|
||||||
return {
|
|
||||||
username: attributes.username,
|
|
||||||
};
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
declare module 'lucia' {
|
|
||||||
interface Register {
|
|
||||||
Lucia: typeof lucia;
|
|
||||||
DatabaseUserAttributes: DatabaseUserAttributes;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
interface DatabaseUserAttributes {
|
|
||||||
username: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
export function getSessionId() {
|
|
||||||
return cookies().get(lucia.sessionCookieName)?.value ?? null;
|
|
||||||
}
|
|
||||||
|
|
||||||
export async function getSession() {
|
|
||||||
const sessionId = getSessionId();
|
|
||||||
if (!sessionId) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
const {session} = await lucia.validateSession(sessionId);
|
|
||||||
return session;
|
|
||||||
}
|
|
||||||
|
|
||||||
export async function getUser() {
|
|
||||||
const sessionId = getSessionId();
|
|
||||||
if (!sessionId) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
const {user, session} = await lucia.validateSession(sessionId);
|
|
||||||
try {
|
|
||||||
if (session && session.fresh) {
|
|
||||||
const sessionCookie = lucia.createSessionCookie(session.id);
|
|
||||||
cookies().set(sessionCookie.name, sessionCookie.value, sessionCookie.attributes);
|
|
||||||
}
|
|
||||||
if (!session) {
|
|
||||||
const sessionCookie = lucia.createBlankSessionCookie();
|
|
||||||
cookies().set(sessionCookie.name, sessionCookie.value, sessionCookie.attributes);
|
|
||||||
}
|
|
||||||
} catch {
|
|
||||||
// Next.js throws error when attempting to set cookies when rendering page
|
|
||||||
}
|
|
||||||
return user;
|
|
||||||
}
|
|
|
@ -1,6 +1,6 @@
|
||||||
'use client';
|
'use client';
|
||||||
|
|
||||||
import { buttonVariants } from '@/components/ui/button';
|
import { Button, buttonVariants } from '@/components/ui/button';
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { Slot } from '@radix-ui/react-slot';
|
import { Slot } from '@radix-ui/react-slot';
|
||||||
import { cn } from '@/lib/utils';
|
import { cn } from '@/lib/utils';
|
||||||
|
@ -25,6 +25,7 @@ export interface ConfirmationDialogProps {
|
||||||
title: string;
|
title: string;
|
||||||
description?: string;
|
description?: string;
|
||||||
actionText?: string;
|
actionText?: string;
|
||||||
|
actionVariant?: 'default' | 'destructive' | 'outline' | 'secondary' | 'ghost' | 'link';
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface ButtonWithActionProps<T = any>
|
export interface ButtonWithActionProps<T = any>
|
||||||
|
@ -76,9 +77,11 @@ const ServerActionTrigger = React.forwardRef<HTMLButtonElement, ButtonWithAction
|
||||||
<AlertDialogCancel>
|
<AlertDialogCancel>
|
||||||
Cancel
|
Cancel
|
||||||
</AlertDialogCancel>
|
</AlertDialogCancel>
|
||||||
<AlertDialogAction onClick={handleSubmit}>
|
<Button variant={props.dialog.actionVariant || 'default'} asChild>
|
||||||
{props.dialog.actionText || 'Confirm'}
|
<AlertDialogAction onClick={handleSubmit}>
|
||||||
</AlertDialogAction>
|
{props.dialog.actionText || 'Confirm'}
|
||||||
|
</AlertDialogAction>
|
||||||
|
</Button>
|
||||||
</AlertDialogFooter>
|
</AlertDialogFooter>
|
||||||
</AlertDialogContent>
|
</AlertDialogContent>
|
||||||
</AlertDialog>
|
</AlertDialog>
|
||||||
|
|
|
@ -1,71 +0,0 @@
|
||||||
'use client';
|
|
||||||
|
|
||||||
import { zodResolver } from '@hookform/resolvers/zod';
|
|
||||||
import { useForm } from 'react-hook-form';
|
|
||||||
import { z } from 'zod';
|
|
||||||
import { Form, FormControl, FormField, FormItem, FormLabel, FormMessage } from '@/components/ui/form';
|
|
||||||
import { Input } from '@/components/ui/input';
|
|
||||||
import React from 'react';
|
|
||||||
import { Button } from '@/components/ui/button';
|
|
||||||
import { signInFormSchema } from '@/lib/form-schemas/signInFormSchema';
|
|
||||||
import { ActionResponse } from '@/lib/types/actionResponse';
|
|
||||||
import { useRouter } from 'next/navigation';
|
|
||||||
import { toast } from 'sonner';
|
|
||||||
import { sonnerContent } from '@/components/ui/sonner';
|
|
||||||
|
|
||||||
export default function SignInForm({onSubmit}: {
|
|
||||||
onSubmit: (data: z.infer<typeof signInFormSchema>) => Promise<ActionResponse>
|
|
||||||
}) {
|
|
||||||
|
|
||||||
const router = useRouter();
|
|
||||||
|
|
||||||
const form = useForm<z.infer<typeof signInFormSchema>>({
|
|
||||||
resolver: zodResolver(signInFormSchema),
|
|
||||||
defaultValues: {
|
|
||||||
username: '',
|
|
||||||
password: '',
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
const handleSubmit = async (data: z.infer<typeof signInFormSchema>) => {
|
|
||||||
const response = await onSubmit(data);
|
|
||||||
toast(sonnerContent(response));
|
|
||||||
if (response.redirect) {
|
|
||||||
router.push(response.redirect);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Form {...form}>
|
|
||||||
<form onSubmit={form.handleSubmit(handleSubmit)} className="space-y-2">
|
|
||||||
<FormField
|
|
||||||
control={form.control}
|
|
||||||
name="username"
|
|
||||||
render={({field}) => (
|
|
||||||
<FormItem>
|
|
||||||
<FormLabel>Username</FormLabel>
|
|
||||||
<FormControl>
|
|
||||||
<Input placeholder="Username" {...field} />
|
|
||||||
</FormControl>
|
|
||||||
<FormMessage/>
|
|
||||||
</FormItem>
|
|
||||||
)}
|
|
||||||
/>
|
|
||||||
<FormField
|
|
||||||
control={form.control}
|
|
||||||
name="password"
|
|
||||||
render={({field}) => (
|
|
||||||
<FormItem>
|
|
||||||
<FormLabel>Password</FormLabel>
|
|
||||||
<FormControl>
|
|
||||||
<Input placeholder="••••••••" type="password" {...field} />
|
|
||||||
</FormControl>
|
|
||||||
<FormMessage/>
|
|
||||||
</FormItem>
|
|
||||||
)}
|
|
||||||
/>
|
|
||||||
<Button type="submit" className="w-full">Sign in</Button>
|
|
||||||
</form>
|
|
||||||
</Form>
|
|
||||||
);
|
|
||||||
}
|
|
|
@ -1,84 +0,0 @@
|
||||||
'use client';
|
|
||||||
|
|
||||||
import { zodResolver } from '@hookform/resolvers/zod';
|
|
||||||
import { useForm } from 'react-hook-form';
|
|
||||||
import { z } from 'zod';
|
|
||||||
import { Form, FormControl, FormField, FormItem, FormLabel, FormMessage } from '@/components/ui/form';
|
|
||||||
import { Input } from '@/components/ui/input';
|
|
||||||
import React from 'react';
|
|
||||||
import { Button } from '@/components/ui/button';
|
|
||||||
import { signUpFormSchema } from '@/lib/form-schemas/signUpFormSchema';
|
|
||||||
import { ActionResponse } from '@/lib/types/actionResponse';
|
|
||||||
import { useRouter } from 'next/navigation';
|
|
||||||
import { toast } from 'sonner';
|
|
||||||
import { sonnerContent } from '@/components/ui/sonner';
|
|
||||||
|
|
||||||
export default function SignUpForm({onSubmit}: {
|
|
||||||
onSubmit: (data: z.infer<typeof signUpFormSchema>) => Promise<ActionResponse>
|
|
||||||
}) {
|
|
||||||
|
|
||||||
const router = useRouter();
|
|
||||||
|
|
||||||
const form = useForm<z.infer<typeof signUpFormSchema>>({
|
|
||||||
resolver: zodResolver(signUpFormSchema),
|
|
||||||
defaultValues: {
|
|
||||||
username: '',
|
|
||||||
password: '',
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
const handleSubmit = async (data: z.infer<typeof signUpFormSchema>) => {
|
|
||||||
const response = await onSubmit(data);
|
|
||||||
toast(sonnerContent(response));
|
|
||||||
if (response.redirect) {
|
|
||||||
router.push(response.redirect);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Form {...form}>
|
|
||||||
<form onSubmit={form.handleSubmit(handleSubmit)} className="space-y-2">
|
|
||||||
<FormField
|
|
||||||
control={form.control}
|
|
||||||
name="username"
|
|
||||||
render={({field}) => (
|
|
||||||
<FormItem>
|
|
||||||
<FormLabel>Username</FormLabel>
|
|
||||||
<FormControl>
|
|
||||||
<Input placeholder="Username" {...field} />
|
|
||||||
</FormControl>
|
|
||||||
<FormMessage/>
|
|
||||||
</FormItem>
|
|
||||||
)}
|
|
||||||
/>
|
|
||||||
<FormField
|
|
||||||
control={form.control}
|
|
||||||
name="password"
|
|
||||||
render={({field}) => (
|
|
||||||
<FormItem>
|
|
||||||
<FormLabel>Password</FormLabel>
|
|
||||||
<FormControl>
|
|
||||||
<Input placeholder="••••••••" type="password" {...field} />
|
|
||||||
</FormControl>
|
|
||||||
<FormMessage/>
|
|
||||||
</FormItem>
|
|
||||||
)}
|
|
||||||
/>
|
|
||||||
<FormField
|
|
||||||
control={form.control}
|
|
||||||
name="confirm"
|
|
||||||
render={({field}) => (
|
|
||||||
<FormItem>
|
|
||||||
<FormLabel>Confirm password</FormLabel>
|
|
||||||
<FormControl>
|
|
||||||
<Input placeholder="••••••••" type="password" {...field} />
|
|
||||||
</FormControl>
|
|
||||||
<FormMessage/>
|
|
||||||
</FormItem>
|
|
||||||
)}
|
|
||||||
/>
|
|
||||||
<Button type="submit" className="w-full">Create Account</Button>
|
|
||||||
</form>
|
|
||||||
</Form>
|
|
||||||
);
|
|
||||||
}
|
|
|
@ -1,58 +0,0 @@
|
||||||
import { ActionResponse } from '@/lib/types/actionResponse';
|
|
||||||
import { URL_SIGN_IN } from '@/lib/constants';
|
|
||||||
import { getUser, lucia } from '@/auth';
|
|
||||||
import prisma from '@/prisma';
|
|
||||||
import { cookies } from 'next/headers';
|
|
||||||
|
|
||||||
export default async function accountDelete(): Promise<ActionResponse> {
|
|
||||||
'use server';
|
|
||||||
|
|
||||||
const user = await getUser();
|
|
||||||
|
|
||||||
if (!user) {
|
|
||||||
return {
|
|
||||||
type: 'error',
|
|
||||||
message: 'You aren\'t signed in.',
|
|
||||||
redirect: URL_SIGN_IN,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
await prisma.payment.deleteMany({
|
|
||||||
where: {
|
|
||||||
userId: user.id,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
await prisma.entity.deleteMany({
|
|
||||||
where: {
|
|
||||||
userId: user.id,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
await prisma.category.deleteMany({
|
|
||||||
where: {
|
|
||||||
userId: user.id,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
await prisma.session.deleteMany({
|
|
||||||
where: {
|
|
||||||
userId: user.id,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
await prisma.user.delete({
|
|
||||||
where: {
|
|
||||||
id: user.id,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
const sessionCookie = lucia.createBlankSessionCookie();
|
|
||||||
cookies().set(sessionCookie.name, sessionCookie.value, sessionCookie.attributes);
|
|
||||||
|
|
||||||
return {
|
|
||||||
type: 'success',
|
|
||||||
message: 'Your account was removed.',
|
|
||||||
redirect: URL_SIGN_IN,
|
|
||||||
};
|
|
||||||
}
|
|
|
@ -1,9 +1,9 @@
|
||||||
import { z } from 'zod';
|
import { z } from 'zod';
|
||||||
import { ActionResponse } from '@/lib/types/actionResponse';
|
import { ActionResponse } from '@/lib/types/actionResponse';
|
||||||
import prisma from '@/prisma';
|
import prisma from '@/prisma';
|
||||||
import { getUser } from '@/auth';
|
|
||||||
import { URL_SIGN_IN } from '@/lib/constants';
|
import { URL_SIGN_IN } from '@/lib/constants';
|
||||||
import { categoryFormSchema } from '@/lib/form-schemas/categoryFormSchema';
|
import { categoryFormSchema } from '@/lib/form-schemas/categoryFormSchema';
|
||||||
|
import { getSession } from '@auth0/nextjs-auth0';
|
||||||
|
|
||||||
export default async function categoryCreateUpdate({
|
export default async function categoryCreateUpdate({
|
||||||
id,
|
id,
|
||||||
|
@ -12,15 +12,15 @@ export default async function categoryCreateUpdate({
|
||||||
}: z.infer<typeof categoryFormSchema>): Promise<ActionResponse> {
|
}: z.infer<typeof categoryFormSchema>): Promise<ActionResponse> {
|
||||||
'use server';
|
'use server';
|
||||||
|
|
||||||
// check that user is logged in
|
const session = await getSession();
|
||||||
const user = await getUser();
|
if (!session) {
|
||||||
if (!user) {
|
|
||||||
return {
|
return {
|
||||||
type: 'error',
|
type: 'error',
|
||||||
message: 'You must be logged in to create/update an category.',
|
message: 'You aren\'t signed in.',
|
||||||
redirect: URL_SIGN_IN,
|
redirect: URL_SIGN_IN,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
const user = session.user;
|
||||||
|
|
||||||
// create/update category
|
// create/update category
|
||||||
try {
|
try {
|
||||||
|
@ -44,7 +44,7 @@ export default async function categoryCreateUpdate({
|
||||||
} else {
|
} else {
|
||||||
await prisma.category.create({
|
await prisma.category.create({
|
||||||
data: {
|
data: {
|
||||||
userId: user.id,
|
userId: user.sub,
|
||||||
name: name,
|
name: name,
|
||||||
color: color,
|
color: color,
|
||||||
},
|
},
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
import { ActionResponse } from '@/lib/types/actionResponse';
|
import { ActionResponse } from '@/lib/types/actionResponse';
|
||||||
import prisma from '@/prisma';
|
import prisma from '@/prisma';
|
||||||
import { getUser } from '@/auth';
|
|
||||||
import { URL_SIGN_IN } from '@/lib/constants';
|
import { URL_SIGN_IN } from '@/lib/constants';
|
||||||
|
import { getSession } from '@auth0/nextjs-auth0';
|
||||||
|
|
||||||
export default async function categoryDelete(id: number): Promise<ActionResponse> {
|
export default async function categoryDelete(id: number): Promise<ActionResponse> {
|
||||||
'use server';
|
'use server';
|
||||||
|
@ -14,21 +14,21 @@ export default async function categoryDelete(id: number): Promise<ActionResponse
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
// check that user is logged in
|
const session = await getSession();
|
||||||
const user = await getUser();
|
if (!session) {
|
||||||
if (!user) {
|
|
||||||
return {
|
return {
|
||||||
type: 'error',
|
type: 'error',
|
||||||
message: 'You must be logged in to delete an category.',
|
message: 'You aren\'t signed in.',
|
||||||
redirect: URL_SIGN_IN,
|
redirect: URL_SIGN_IN,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
const user = session.user;
|
||||||
|
|
||||||
// check that category is associated with user
|
// check that category is associated with user
|
||||||
const category = await prisma.category.findFirst({
|
const category = await prisma.category.findFirst({
|
||||||
where: {
|
where: {
|
||||||
id: id,
|
id: id,
|
||||||
userId: user.id,
|
userId: user.sub,
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
if (!category) {
|
if (!category) {
|
||||||
|
@ -43,7 +43,7 @@ export default async function categoryDelete(id: number): Promise<ActionResponse
|
||||||
await prisma.category.delete({
|
await prisma.category.delete({
|
||||||
where: {
|
where: {
|
||||||
id: category.id,
|
id: category.id,
|
||||||
userId: user.id,
|
userId: user.sub,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
|
|
40
src/lib/actions/clearAccountData.ts
Normal file
40
src/lib/actions/clearAccountData.ts
Normal file
|
@ -0,0 +1,40 @@
|
||||||
|
import { ActionResponse } from '@/lib/types/actionResponse';
|
||||||
|
import { URL_SIGN_IN } from '@/lib/constants';
|
||||||
|
import prisma from '@/prisma';
|
||||||
|
import { getSession } from '@auth0/nextjs-auth0';
|
||||||
|
|
||||||
|
export default async function clearAccountData(): Promise<ActionResponse> {
|
||||||
|
'use server';
|
||||||
|
|
||||||
|
const session = await getSession();
|
||||||
|
if (!session) {
|
||||||
|
return {
|
||||||
|
type: 'error',
|
||||||
|
message: 'You aren\'t signed in.',
|
||||||
|
redirect: URL_SIGN_IN,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
await prisma.payment.deleteMany({
|
||||||
|
where: {
|
||||||
|
userId: session.user.sub,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
await prisma.entity.deleteMany({
|
||||||
|
where: {
|
||||||
|
userId: session.user.sub,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
await prisma.category.deleteMany({
|
||||||
|
where: {
|
||||||
|
userId: session.user.sub,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
return {
|
||||||
|
type: 'success',
|
||||||
|
message: 'Your account data was cleared.',
|
||||||
|
};
|
||||||
|
}
|
|
@ -2,8 +2,8 @@ import { z } from 'zod';
|
||||||
import { ActionResponse } from '@/lib/types/actionResponse';
|
import { ActionResponse } from '@/lib/types/actionResponse';
|
||||||
import { entityFormSchema } from '@/lib/form-schemas/entityFormSchema';
|
import { entityFormSchema } from '@/lib/form-schemas/entityFormSchema';
|
||||||
import prisma from '@/prisma';
|
import prisma from '@/prisma';
|
||||||
import { getUser } from '@/auth';
|
|
||||||
import { URL_SIGN_IN } from '@/lib/constants';
|
import { URL_SIGN_IN } from '@/lib/constants';
|
||||||
|
import { getSession } from '@auth0/nextjs-auth0';
|
||||||
|
|
||||||
export default async function entityCreateUpdate({
|
export default async function entityCreateUpdate({
|
||||||
id,
|
id,
|
||||||
|
@ -13,15 +13,15 @@ export default async function entityCreateUpdate({
|
||||||
}: z.infer<typeof entityFormSchema>): Promise<ActionResponse> {
|
}: z.infer<typeof entityFormSchema>): Promise<ActionResponse> {
|
||||||
'use server';
|
'use server';
|
||||||
|
|
||||||
// check that user is logged in
|
const session = await getSession();
|
||||||
const user = await getUser();
|
if (!session) {
|
||||||
if (!user) {
|
|
||||||
return {
|
return {
|
||||||
type: 'error',
|
type: 'error',
|
||||||
message: 'You must be logged in to create/update an entity.',
|
message: 'You aren\'t signed in.',
|
||||||
redirect: URL_SIGN_IN,
|
redirect: URL_SIGN_IN,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
const user = session.user;
|
||||||
|
|
||||||
// create/update entity
|
// create/update entity
|
||||||
try {
|
try {
|
||||||
|
@ -46,7 +46,7 @@ export default async function entityCreateUpdate({
|
||||||
} else {
|
} else {
|
||||||
await prisma.entity.create({
|
await prisma.entity.create({
|
||||||
data: {
|
data: {
|
||||||
userId: user.id,
|
userId: user.sub,
|
||||||
name: name,
|
name: name,
|
||||||
type: type,
|
type: type,
|
||||||
defaultCategoryId: defaultCategoryId ?? null,
|
defaultCategoryId: defaultCategoryId ?? null,
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
import { ActionResponse } from '@/lib/types/actionResponse';
|
import { ActionResponse } from '@/lib/types/actionResponse';
|
||||||
import prisma from '@/prisma';
|
import prisma from '@/prisma';
|
||||||
import { getUser } from '@/auth';
|
|
||||||
import { URL_SIGN_IN } from '@/lib/constants';
|
import { URL_SIGN_IN } from '@/lib/constants';
|
||||||
|
import { getSession } from '@auth0/nextjs-auth0';
|
||||||
|
|
||||||
export default async function entityDelete(id: number): Promise<ActionResponse> {
|
export default async function entityDelete(id: number): Promise<ActionResponse> {
|
||||||
'use server';
|
'use server';
|
||||||
|
@ -14,21 +14,21 @@ export default async function entityDelete(id: number): Promise<ActionResponse>
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
// check that user is logged in
|
const session = await getSession();
|
||||||
const user = await getUser();
|
if (!session) {
|
||||||
if (!user) {
|
|
||||||
return {
|
return {
|
||||||
type: 'error',
|
type: 'error',
|
||||||
message: 'You must be logged in to delete an entity.',
|
message: 'You aren\'t signed in.',
|
||||||
redirect: URL_SIGN_IN,
|
redirect: URL_SIGN_IN,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
const user = session.user;
|
||||||
|
|
||||||
// check that entity is associated with user
|
// check that entity is associated with user
|
||||||
const entity = await prisma.entity.findFirst({
|
const entity = await prisma.entity.findFirst({
|
||||||
where: {
|
where: {
|
||||||
id: id,
|
id: id,
|
||||||
userId: user.id,
|
userId: user.sub,
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
if (!entity) {
|
if (!entity) {
|
||||||
|
@ -43,7 +43,7 @@ export default async function entityDelete(id: number): Promise<ActionResponse>
|
||||||
await prisma.entity.delete({
|
await prisma.entity.delete({
|
||||||
where: {
|
where: {
|
||||||
id: entity.id,
|
id: entity.id,
|
||||||
userId: user.id,
|
userId: user.sub,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
|
|
|
@ -1,32 +1,32 @@
|
||||||
import prisma from '@/prisma';
|
import prisma from '@/prisma';
|
||||||
import type { Category, Entity } from '@prisma/client';
|
import type { Category, Entity } from '@prisma/client';
|
||||||
import { EntityType } from '@prisma/client';
|
import { EntityType } from '@prisma/client';
|
||||||
import { getUser } from '@/auth';
|
|
||||||
import { URL_SIGN_IN } from '@/lib/constants';
|
import { URL_SIGN_IN } from '@/lib/constants';
|
||||||
import { ActionResponse } from '@/lib/types/actionResponse';
|
import { ActionResponse } from '@/lib/types/actionResponse';
|
||||||
|
import { getSession } from '@auth0/nextjs-auth0';
|
||||||
|
|
||||||
export default async function generateSampleData(): Promise<ActionResponse> {
|
export default async function generateSampleData(): Promise<ActionResponse> {
|
||||||
'use server';
|
'use server';
|
||||||
|
|
||||||
const user = await getUser();
|
const session = await getSession();
|
||||||
|
if (!session) {
|
||||||
if (!user) {
|
|
||||||
return {
|
return {
|
||||||
type: 'error',
|
type: 'error',
|
||||||
message: 'You must be logged in to create/update an category.',
|
message: 'You aren\'t signed in.',
|
||||||
redirect: URL_SIGN_IN,
|
redirect: URL_SIGN_IN,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
const user = session.user;
|
||||||
|
|
||||||
// Categories: create sample data
|
// Categories: create sample data
|
||||||
const categories: Category[] = await prisma.category.findMany({where: {userId: user.id}});
|
const categories: Category[] = await prisma.category.findMany({where: {userId: user.sub}});
|
||||||
if (await prisma.category.count({where: {userId: user.id}}) == 0) {
|
if (await prisma.category.count({where: {userId: user.sub}}) == 0) {
|
||||||
|
|
||||||
console.log('Creating sample categories...');
|
console.log('Creating sample categories...');
|
||||||
|
|
||||||
categories.push(await prisma.category.create({
|
categories.push(await prisma.category.create({
|
||||||
data: {
|
data: {
|
||||||
userId: user.id,
|
userId: user.sub,
|
||||||
name: 'Groceries',
|
name: 'Groceries',
|
||||||
color: '#FFBEAC',
|
color: '#FFBEAC',
|
||||||
},
|
},
|
||||||
|
@ -34,7 +34,7 @@ export default async function generateSampleData(): Promise<ActionResponse> {
|
||||||
|
|
||||||
categories.push(await prisma.category.create({
|
categories.push(await prisma.category.create({
|
||||||
data: {
|
data: {
|
||||||
userId: user.id,
|
userId: user.sub,
|
||||||
name: 'Drugstore items',
|
name: 'Drugstore items',
|
||||||
color: '#9CBCFF',
|
color: '#9CBCFF',
|
||||||
},
|
},
|
||||||
|
@ -42,7 +42,7 @@ export default async function generateSampleData(): Promise<ActionResponse> {
|
||||||
|
|
||||||
categories.push(await prisma.category.create({
|
categories.push(await prisma.category.create({
|
||||||
data: {
|
data: {
|
||||||
userId: user.id,
|
userId: user.sub,
|
||||||
name: 'Going out',
|
name: 'Going out',
|
||||||
color: '#F1ADFF',
|
color: '#F1ADFF',
|
||||||
},
|
},
|
||||||
|
@ -50,7 +50,7 @@ export default async function generateSampleData(): Promise<ActionResponse> {
|
||||||
|
|
||||||
categories.push(await prisma.category.create({
|
categories.push(await prisma.category.create({
|
||||||
data: {
|
data: {
|
||||||
userId: user.id,
|
userId: user.sub,
|
||||||
name: 'Random stuff',
|
name: 'Random stuff',
|
||||||
color: '#C1FFA9',
|
color: '#C1FFA9',
|
||||||
},
|
},
|
||||||
|
@ -58,7 +58,7 @@ export default async function generateSampleData(): Promise<ActionResponse> {
|
||||||
|
|
||||||
categories.push(await prisma.category.create({
|
categories.push(await prisma.category.create({
|
||||||
data: {
|
data: {
|
||||||
userId: user.id,
|
userId: user.sub,
|
||||||
name: 'Salary',
|
name: 'Salary',
|
||||||
color: '#FFF787',
|
color: '#FFF787',
|
||||||
},
|
},
|
||||||
|
@ -69,14 +69,14 @@ export default async function generateSampleData(): Promise<ActionResponse> {
|
||||||
console.log(categories);
|
console.log(categories);
|
||||||
|
|
||||||
// Entities: create sample data
|
// Entities: create sample data
|
||||||
const entities: Entity[] = await prisma.entity.findMany({where: {userId: user.id}});
|
const entities: Entity[] = await prisma.entity.findMany({where: {userId: user.sub}});
|
||||||
if (await prisma.entity.count({where: {userId: user.id}}) == 0) {
|
if (await prisma.entity.count({where: {userId: user.sub}}) == 0) {
|
||||||
|
|
||||||
console.log('Creating sample entities...');
|
console.log('Creating sample entities...');
|
||||||
|
|
||||||
entities.push(await prisma.entity.create({
|
entities.push(await prisma.entity.create({
|
||||||
data: {
|
data: {
|
||||||
userId: user.id,
|
userId: user.sub,
|
||||||
name: 'Main Account',
|
name: 'Main Account',
|
||||||
type: EntityType.Account,
|
type: EntityType.Account,
|
||||||
},
|
},
|
||||||
|
@ -84,7 +84,7 @@ export default async function generateSampleData(): Promise<ActionResponse> {
|
||||||
|
|
||||||
entities.push(await prisma.entity.create({
|
entities.push(await prisma.entity.create({
|
||||||
data: {
|
data: {
|
||||||
userId: user.id,
|
userId: user.sub,
|
||||||
name: 'Company',
|
name: 'Company',
|
||||||
type: EntityType.Entity,
|
type: EntityType.Entity,
|
||||||
},
|
},
|
||||||
|
@ -92,7 +92,7 @@ export default async function generateSampleData(): Promise<ActionResponse> {
|
||||||
|
|
||||||
entities.push(await prisma.entity.create({
|
entities.push(await prisma.entity.create({
|
||||||
data: {
|
data: {
|
||||||
userId: user.id,
|
userId: user.sub,
|
||||||
name: 'Supermarket 1',
|
name: 'Supermarket 1',
|
||||||
type: EntityType.Entity,
|
type: EntityType.Entity,
|
||||||
},
|
},
|
||||||
|
@ -100,7 +100,7 @@ export default async function generateSampleData(): Promise<ActionResponse> {
|
||||||
|
|
||||||
entities.push(await prisma.entity.create({
|
entities.push(await prisma.entity.create({
|
||||||
data: {
|
data: {
|
||||||
userId: user.id,
|
userId: user.sub,
|
||||||
name: 'Supermarket 2',
|
name: 'Supermarket 2',
|
||||||
type: EntityType.Entity,
|
type: EntityType.Entity,
|
||||||
},
|
},
|
||||||
|
@ -108,7 +108,7 @@ export default async function generateSampleData(): Promise<ActionResponse> {
|
||||||
|
|
||||||
entities.push(await prisma.entity.create({
|
entities.push(await prisma.entity.create({
|
||||||
data: {
|
data: {
|
||||||
userId: user.id,
|
userId: user.sub,
|
||||||
name: 'Supermarket 3',
|
name: 'Supermarket 3',
|
||||||
type: EntityType.Entity,
|
type: EntityType.Entity,
|
||||||
},
|
},
|
||||||
|
@ -116,7 +116,7 @@ export default async function generateSampleData(): Promise<ActionResponse> {
|
||||||
|
|
||||||
entities.push(await prisma.entity.create({
|
entities.push(await prisma.entity.create({
|
||||||
data: {
|
data: {
|
||||||
userId: user.id,
|
userId: user.sub,
|
||||||
name: 'Supermarket 4',
|
name: 'Supermarket 4',
|
||||||
type: EntityType.Entity,
|
type: EntityType.Entity,
|
||||||
},
|
},
|
||||||
|
@ -129,21 +129,24 @@ export default async function generateSampleData(): Promise<ActionResponse> {
|
||||||
// Payments: create sample data
|
// Payments: create sample data
|
||||||
console.log('Creating sample payments...');
|
console.log('Creating sample payments...');
|
||||||
|
|
||||||
if (await prisma.payment.count({where: {userId: user.id}}) == 0) {
|
if (await prisma.payment.count({where: {userId: user.sub}}) == 0) {
|
||||||
for (let i = 0; i < 4; i++) {
|
for (let i = 0; i < 4; i++) {
|
||||||
|
|
||||||
const date = new Date();
|
const date = new Date();
|
||||||
date.setDate(1);
|
date.setDate(1);
|
||||||
date.setMonth(date.getMonth() - i);
|
date.setMonth(date.getMonth() - i);
|
||||||
|
|
||||||
|
const categoryId =
|
||||||
|
categories.find((it) => it.name === 'Salary')?.id!;
|
||||||
|
|
||||||
await prisma.payment.create({
|
await prisma.payment.create({
|
||||||
data: {
|
data: {
|
||||||
userId: user.id,
|
userId: user.sub,
|
||||||
amount: 200000,
|
amount: 200000,
|
||||||
date: date,
|
date: date,
|
||||||
payorId: entities[1].id,
|
payorId: entities[1].id,
|
||||||
payeeId: entities[0].id,
|
payeeId: entities[0].id,
|
||||||
categoryId: 5,
|
categoryId: categoryId,
|
||||||
createdAt: date,
|
createdAt: date,
|
||||||
updatedAt: date,
|
updatedAt: date,
|
||||||
},
|
},
|
||||||
|
@ -166,7 +169,7 @@ export default async function generateSampleData(): Promise<ActionResponse> {
|
||||||
|
|
||||||
await prisma.payment.create({
|
await prisma.payment.create({
|
||||||
data: {
|
data: {
|
||||||
userId: user.id,
|
userId: user.sub,
|
||||||
amount: Math.floor(
|
amount: Math.floor(
|
||||||
Math.random() * (maxAmount - minAmount) + minAmount),
|
Math.random() * (maxAmount - minAmount) + minAmount),
|
||||||
date: date,
|
date: date,
|
||||||
|
|
|
@ -1,9 +1,9 @@
|
||||||
import { z } from 'zod';
|
import { z } from 'zod';
|
||||||
import { ActionResponse } from '@/lib/types/actionResponse';
|
import { ActionResponse } from '@/lib/types/actionResponse';
|
||||||
import prisma from '@/prisma';
|
import prisma from '@/prisma';
|
||||||
import { getUser } from '@/auth';
|
|
||||||
import { URL_SIGN_IN } from '@/lib/constants';
|
import { URL_SIGN_IN } from '@/lib/constants';
|
||||||
import { paymentFormSchema } from '@/lib/form-schemas/paymentFormSchema';
|
import { paymentFormSchema } from '@/lib/form-schemas/paymentFormSchema';
|
||||||
|
import { getSession } from '@auth0/nextjs-auth0';
|
||||||
|
|
||||||
export default async function paymentCreateUpdate({
|
export default async function paymentCreateUpdate({
|
||||||
id,
|
id,
|
||||||
|
@ -16,15 +16,15 @@ export default async function paymentCreateUpdate({
|
||||||
}: z.infer<typeof paymentFormSchema>): Promise<ActionResponse> {
|
}: z.infer<typeof paymentFormSchema>): Promise<ActionResponse> {
|
||||||
'use server';
|
'use server';
|
||||||
|
|
||||||
// check that user is logged in
|
const session = await getSession();
|
||||||
const user = await getUser();
|
if (!session) {
|
||||||
if (!user) {
|
|
||||||
return {
|
return {
|
||||||
type: 'error',
|
type: 'error',
|
||||||
message: 'You must be logged in to create/update a payment.',
|
message: 'You aren\'t signed in.',
|
||||||
redirect: URL_SIGN_IN,
|
redirect: URL_SIGN_IN,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
const user = session.user;
|
||||||
|
|
||||||
// create/update payment
|
// create/update payment
|
||||||
try {
|
try {
|
||||||
|
@ -52,7 +52,7 @@ export default async function paymentCreateUpdate({
|
||||||
} else {
|
} else {
|
||||||
await prisma.payment.create({
|
await prisma.payment.create({
|
||||||
data: {
|
data: {
|
||||||
userId: user.id,
|
userId: user.sub,
|
||||||
amount: amount,
|
amount: amount,
|
||||||
date: date,
|
date: date,
|
||||||
payorId: payorId,
|
payorId: payorId,
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
import { ActionResponse } from '@/lib/types/actionResponse';
|
import { ActionResponse } from '@/lib/types/actionResponse';
|
||||||
import prisma from '@/prisma';
|
import prisma from '@/prisma';
|
||||||
import { getUser } from '@/auth';
|
|
||||||
import { URL_SIGN_IN } from '@/lib/constants';
|
import { URL_SIGN_IN } from '@/lib/constants';
|
||||||
|
import { getSession } from '@auth0/nextjs-auth0';
|
||||||
|
|
||||||
export default async function paymentDelete(id: number): Promise<ActionResponse> {
|
export default async function paymentDelete(id: number): Promise<ActionResponse> {
|
||||||
'use server';
|
'use server';
|
||||||
|
@ -14,21 +14,21 @@ export default async function paymentDelete(id: number): Promise<ActionResponse>
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
// check that user is logged in
|
const session = await getSession();
|
||||||
const user = await getUser();
|
if (!session) {
|
||||||
if (!user) {
|
|
||||||
return {
|
return {
|
||||||
type: 'error',
|
type: 'error',
|
||||||
message: 'You must be logged in to delete a payment.',
|
message: 'You aren\'t signed in.',
|
||||||
redirect: URL_SIGN_IN,
|
redirect: URL_SIGN_IN,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
const user = session.user;
|
||||||
|
|
||||||
// check that payment is associated with user
|
// check that payment is associated with user
|
||||||
const payment = await prisma.payment.findFirst({
|
const payment = await prisma.payment.findFirst({
|
||||||
where: {
|
where: {
|
||||||
id: id,
|
id: id,
|
||||||
userId: user.id,
|
userId: user.sub,
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
if (!payment) {
|
if (!payment) {
|
||||||
|
@ -43,7 +43,7 @@ export default async function paymentDelete(id: number): Promise<ActionResponse>
|
||||||
await prisma.payment.delete({
|
await prisma.payment.delete({
|
||||||
where: {
|
where: {
|
||||||
id: payment.id,
|
id: payment.id,
|
||||||
userId: user.id,
|
userId: user.sub,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
|
|
|
@ -1,41 +0,0 @@
|
||||||
import { z } from 'zod';
|
|
||||||
import { Argon2id } from 'oslo/password';
|
|
||||||
import { lucia } from '@/auth';
|
|
||||||
import { cookies } from 'next/headers';
|
|
||||||
import { signInFormSchema } from '@/lib/form-schemas/signInFormSchema';
|
|
||||||
import { ActionResponse } from '@/lib/types/actionResponse';
|
|
||||||
import { URL_HOME } from '@/lib/constants';
|
|
||||||
import prisma from '@/prisma';
|
|
||||||
|
|
||||||
export default async function signIn({username, password}: z.infer<typeof signInFormSchema>): Promise<ActionResponse> {
|
|
||||||
'use server';
|
|
||||||
|
|
||||||
const existingUser = await prisma.user.findFirst({
|
|
||||||
where: {
|
|
||||||
username: username.toLowerCase(),
|
|
||||||
},
|
|
||||||
});
|
|
||||||
if (!existingUser) {
|
|
||||||
return {
|
|
||||||
type: 'error',
|
|
||||||
message: 'Incorrect username or password',
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
const validPassword = await new Argon2id().verify(existingUser.password, password);
|
|
||||||
if (!validPassword) {
|
|
||||||
return {
|
|
||||||
type: 'error',
|
|
||||||
message: 'Incorrect username or password',
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
const session = await lucia.createSession(existingUser.id, {});
|
|
||||||
const sessionCookie = lucia.createSessionCookie(session.id);
|
|
||||||
cookies().set(sessionCookie.name, sessionCookie.value, sessionCookie.attributes);
|
|
||||||
return {
|
|
||||||
type: 'success',
|
|
||||||
message: 'Signed in successfully',
|
|
||||||
redirect: URL_HOME,
|
|
||||||
};
|
|
||||||
}
|
|
|
@ -1,26 +0,0 @@
|
||||||
import { getSession, lucia } from '@/auth';
|
|
||||||
import { cookies } from 'next/headers';
|
|
||||||
import { ActionResponse } from '@/lib/types/actionResponse';
|
|
||||||
import { URL_SIGN_IN } from '@/lib/constants';
|
|
||||||
|
|
||||||
export default async function signOut(): Promise<ActionResponse> {
|
|
||||||
'use server';
|
|
||||||
|
|
||||||
const session = await getSession();
|
|
||||||
if (!session) {
|
|
||||||
return {
|
|
||||||
type: 'error',
|
|
||||||
message: 'You aren\'t signed in',
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
await lucia.invalidateSession(session.id);
|
|
||||||
|
|
||||||
const sessionCookie = lucia.createBlankSessionCookie();
|
|
||||||
cookies().set(sessionCookie.name, sessionCookie.value, sessionCookie.attributes);
|
|
||||||
return {
|
|
||||||
type: 'success',
|
|
||||||
message: 'Signed out successfully',
|
|
||||||
redirect: URL_SIGN_IN,
|
|
||||||
};
|
|
||||||
}
|
|
|
@ -1,46 +0,0 @@
|
||||||
import { z } from 'zod';
|
|
||||||
import { Argon2id } from 'oslo/password';
|
|
||||||
import { generateId } from 'lucia';
|
|
||||||
import { lucia } from '@/auth';
|
|
||||||
import { cookies } from 'next/headers';
|
|
||||||
import { signUpFormSchema } from '@/lib/form-schemas/signUpFormSchema';
|
|
||||||
import { ActionResponse } from '@/lib/types/actionResponse';
|
|
||||||
import { URL_HOME } from '@/lib/constants';
|
|
||||||
import prisma from '@/prisma';
|
|
||||||
|
|
||||||
export default async function signUp({username, password}: z.infer<typeof signUpFormSchema>): Promise<ActionResponse> {
|
|
||||||
'use server';
|
|
||||||
|
|
||||||
const hashedPassword = await new Argon2id().hash(password);
|
|
||||||
const userId = generateId(15);
|
|
||||||
|
|
||||||
const existingUser = await prisma.user.findFirst({
|
|
||||||
where: {
|
|
||||||
username: username.toLowerCase(),
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
if (existingUser) {
|
|
||||||
return {
|
|
||||||
type: 'error',
|
|
||||||
message: 'Username already exists',
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
await prisma.user.create({
|
|
||||||
data: {
|
|
||||||
id: userId,
|
|
||||||
username: username,
|
|
||||||
password: hashedPassword,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
const session = await lucia.createSession(userId, {});
|
|
||||||
const sessionCookie = lucia.createSessionCookie(session.id);
|
|
||||||
cookies().set(sessionCookie.name, sessionCookie.value, sessionCookie.attributes);
|
|
||||||
return {
|
|
||||||
type: 'success',
|
|
||||||
message: 'Signed up successfully',
|
|
||||||
redirect: URL_HOME,
|
|
||||||
};
|
|
||||||
}
|
|
|
@ -1,7 +1,6 @@
|
||||||
// auth urls
|
export const URL_SIGN_IN = `/api/auth/login`;
|
||||||
export const URL_AUTH = '/auth';
|
export const URL_SIGN_OUT = `/api/auth/logout`;
|
||||||
export const URL_SIGN_IN = `${URL_AUTH}/signin`;
|
|
||||||
export const URL_SIGN_UP = `${URL_AUTH}/signup`;
|
|
||||||
|
|
||||||
// main urls
|
// main urls
|
||||||
export const URL_HOME = '/';
|
export const URL_HOME = '/';
|
||||||
|
|
|
@ -1,6 +0,0 @@
|
||||||
import { z } from 'zod';
|
|
||||||
|
|
||||||
export const signInFormSchema = z.object({
|
|
||||||
username: z.string().min(3).max(16),
|
|
||||||
password: z.string().min(8).max(255),
|
|
||||||
});
|
|
|
@ -1,10 +0,0 @@
|
||||||
import { z } from 'zod';
|
|
||||||
|
|
||||||
export const signUpFormSchema = z.object({
|
|
||||||
username: z.string().min(3).max(16),
|
|
||||||
password: z.string().min(8).max(255),
|
|
||||||
confirm: z.string().min(8).max(255),
|
|
||||||
}).refine(data => data.password === data.confirm, {
|
|
||||||
message: 'Passwords do not match',
|
|
||||||
path: ['confirm'],
|
|
||||||
});
|
|
|
@ -1,27 +1,3 @@
|
||||||
import type { NextRequest } from 'next/server';
|
import { withMiddlewareAuthRequired } from '@auth0/nextjs-auth0/edge';
|
||||||
import { NextResponse } from 'next/server';
|
|
||||||
import { URL_AUTH, URL_HOME, URL_SIGN_IN } from './lib/constants';
|
|
||||||
|
|
||||||
export async function middleware(request: NextRequest) {
|
export default withMiddlewareAuthRequired();
|
||||||
|
|
||||||
// get session id from cookies
|
|
||||||
const sessionId = request.cookies.get('auth_session')?.value ?? null;
|
|
||||||
|
|
||||||
// redirect to home if user is already authenticated
|
|
||||||
if (request.nextUrl.pathname.startsWith(URL_AUTH) && sessionId) {
|
|
||||||
return NextResponse.redirect(new URL(URL_HOME, request.url));
|
|
||||||
}
|
|
||||||
|
|
||||||
// redirect to sign in if user is not authenticated
|
|
||||||
if (!request.nextUrl.pathname.startsWith(URL_AUTH) && !sessionId) {
|
|
||||||
return NextResponse.redirect(new URL(URL_SIGN_IN, request.url));
|
|
||||||
}
|
|
||||||
|
|
||||||
return NextResponse.next();
|
|
||||||
}
|
|
||||||
|
|
||||||
export const config = {
|
|
||||||
matcher: [
|
|
||||||
'/((?!api|_next/static|_next/image|favicon.ico).*)',
|
|
||||||
],
|
|
||||||
};
|
|
||||||
|
|
Loading…
Add table
Reference in a new issue