🗑️ Clean up code
This commit is contained in:
parent
7a072988ce
commit
142e7c3434
@ -1,6 +1,6 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<project version="4">
|
<project version="4">
|
||||||
<component name="dataSourceStorageLocal" created-in="GO-241.18034.61">
|
<component name="dataSourceStorageLocal" created-in="GO-242.20224.306">
|
||||||
<data-source name="hy_passport@localhost" uuid="74bcf3ef-a2b9-435b-b9e5-f32902a33b25">
|
<data-source name="hy_passport@localhost" uuid="74bcf3ef-a2b9-435b-b9e5-f32902a33b25">
|
||||||
<database-info product="PostgreSQL" version="16.3 (Homebrew)" jdbc-version="4.2" driver-name="PostgreSQL JDBC Driver" driver-version="42.6.0" dbms="POSTGRES" exact-version="16.3" exact-driver-version="42.6">
|
<database-info product="PostgreSQL" version="16.3 (Homebrew)" jdbc-version="4.2" driver-name="PostgreSQL JDBC Driver" driver-version="42.6.0" dbms="POSTGRES" exact-version="16.3" exact-driver-version="42.6">
|
||||||
<identifier-quote-string>"</identifier-quote-string>
|
<identifier-quote-string>"</identifier-quote-string>
|
||||||
|
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
@ -4,7 +4,66 @@
|
|||||||
<option name="autoReloadType" value="ALL" />
|
<option name="autoReloadType" value="ALL" />
|
||||||
</component>
|
</component>
|
||||||
<component name="ChangeListManager">
|
<component name="ChangeListManager">
|
||||||
<list default="true" id="3fefb2c4-b6f9-466b-a523-53352e8d6f95" name="更改" comment=":bug: Fix permissions in groups" />
|
<list default="true" id="3fefb2c4-b6f9-466b-a523-53352e8d6f95" name="更改" comment=":bug: Fix permissions in groups">
|
||||||
|
<change beforePath="$PROJECT_DIR$/.idea/dataSources.local.xml" beforeDir="false" afterPath="$PROJECT_DIR$/.idea/dataSources.local.xml" afterDir="false" />
|
||||||
|
<change beforePath="$PROJECT_DIR$/.idea/dataSources/723637bc-6ce3-4bbe-afb3-d88730c75a1b.xml" beforeDir="false" afterPath="$PROJECT_DIR$/.idea/dataSources/723637bc-6ce3-4bbe-afb3-d88730c75a1b.xml" afterDir="false" />
|
||||||
|
<change beforePath="$PROJECT_DIR$/.idea/dataSources/74bcf3ef-a2b9-435b-b9e5-f32902a33b25.xml" beforeDir="false" afterPath="$PROJECT_DIR$/.idea/dataSources/74bcf3ef-a2b9-435b-b9e5-f32902a33b25.xml" afterDir="false" />
|
||||||
|
<change beforePath="$PROJECT_DIR$/.idea/workspace.xml" beforeDir="false" afterPath="$PROJECT_DIR$/.idea/workspace.xml" afterDir="false" />
|
||||||
|
<change beforePath="$PROJECT_DIR$/Dockerfile" beforeDir="false" afterPath="$PROJECT_DIR$/Dockerfile" afterDir="false" />
|
||||||
|
<change beforePath="$PROJECT_DIR$/go.mod" beforeDir="false" afterPath="$PROJECT_DIR$/go.mod" afterDir="false" />
|
||||||
|
<change beforePath="$PROJECT_DIR$/go.sum" beforeDir="false" afterPath="$PROJECT_DIR$/go.sum" afterDir="false" />
|
||||||
|
<change beforePath="$PROJECT_DIR$/settings.toml" beforeDir="false" afterPath="$PROJECT_DIR$/settings.toml" afterDir="false" />
|
||||||
|
<change beforePath="$PROJECT_DIR$/web/.eslintrc.cjs" beforeDir="false" />
|
||||||
|
<change beforePath="$PROJECT_DIR$/web/.gitignore" beforeDir="false" />
|
||||||
|
<change beforePath="$PROJECT_DIR$/web/.prettierrc.json" beforeDir="false" />
|
||||||
|
<change beforePath="$PROJECT_DIR$/web/.vite/deps/_metadata.json" beforeDir="false" />
|
||||||
|
<change beforePath="$PROJECT_DIR$/web/.vite/deps/package.json" beforeDir="false" />
|
||||||
|
<change beforePath="$PROJECT_DIR$/web/README.md" beforeDir="false" />
|
||||||
|
<change beforePath="$PROJECT_DIR$/web/bun.lockb" beforeDir="false" />
|
||||||
|
<change beforePath="$PROJECT_DIR$/web/env.d.ts" beforeDir="false" />
|
||||||
|
<change beforePath="$PROJECT_DIR$/web/index.html" beforeDir="false" />
|
||||||
|
<change beforePath="$PROJECT_DIR$/web/package.json" beforeDir="false" />
|
||||||
|
<change beforePath="$PROJECT_DIR$/web/public/favicon.png" beforeDir="false" />
|
||||||
|
<change beforePath="$PROJECT_DIR$/web/src/assets/utils.css" beforeDir="false" />
|
||||||
|
<change beforePath="$PROJECT_DIR$/web/src/components/Copyright.vue" beforeDir="false" />
|
||||||
|
<change beforePath="$PROJECT_DIR$/web/src/components/GoUseSolian.vue" beforeDir="false" />
|
||||||
|
<change beforePath="$PROJECT_DIR$/web/src/components/NotificationList.vue" beforeDir="false" />
|
||||||
|
<change beforePath="$PROJECT_DIR$/web/src/components/UserMenu.vue" beforeDir="false" />
|
||||||
|
<change beforePath="$PROJECT_DIR$/web/src/components/admin/UserAssignPermsPanel.vue" beforeDir="false" />
|
||||||
|
<change beforePath="$PROJECT_DIR$/web/src/components/admin/UserDetailPanel.vue" beforeDir="false" />
|
||||||
|
<change beforePath="$PROJECT_DIR$/web/src/components/admin/UserFactorPanel.vue" beforeDir="false" />
|
||||||
|
<change beforePath="$PROJECT_DIR$/web/src/components/auth/Authenticate.vue" beforeDir="false" />
|
||||||
|
<change beforePath="$PROJECT_DIR$/web/src/components/auth/AuthenticateCompleted.vue" beforeDir="false" />
|
||||||
|
<change beforePath="$PROJECT_DIR$/web/src/components/auth/CallbackNotify.vue" beforeDir="false" />
|
||||||
|
<change beforePath="$PROJECT_DIR$/web/src/components/auth/FactorApplicator.vue" beforeDir="false" />
|
||||||
|
<change beforePath="$PROJECT_DIR$/web/src/components/auth/FactorPicker.vue" beforeDir="false" />
|
||||||
|
<change beforePath="$PROJECT_DIR$/web/src/components/navigation/AppBar.vue" beforeDir="false" />
|
||||||
|
<change beforePath="$PROJECT_DIR$/web/src/index.vue" beforeDir="false" />
|
||||||
|
<change beforePath="$PROJECT_DIR$/web/src/layouts/administrator.vue" beforeDir="false" />
|
||||||
|
<change beforePath="$PROJECT_DIR$/web/src/layouts/master.vue" beforeDir="false" />
|
||||||
|
<change beforePath="$PROJECT_DIR$/web/src/layouts/user-center.vue" beforeDir="false" />
|
||||||
|
<change beforePath="$PROJECT_DIR$/web/src/main.ts" beforeDir="false" />
|
||||||
|
<change beforePath="$PROJECT_DIR$/web/src/router/index.ts" beforeDir="false" />
|
||||||
|
<change beforePath="$PROJECT_DIR$/web/src/scripts/request.ts" beforeDir="false" />
|
||||||
|
<change beforePath="$PROJECT_DIR$/web/src/stores/notifications.ts" beforeDir="false" />
|
||||||
|
<change beforePath="$PROJECT_DIR$/web/src/stores/userinfo.ts" beforeDir="false" />
|
||||||
|
<change beforePath="$PROJECT_DIR$/web/src/views/admin/dashboard.vue" beforeDir="false" />
|
||||||
|
<change beforePath="$PROJECT_DIR$/web/src/views/admin/users.vue" beforeDir="false" />
|
||||||
|
<change beforePath="$PROJECT_DIR$/web/src/views/auth/authorize.vue" beforeDir="false" />
|
||||||
|
<change beforePath="$PROJECT_DIR$/web/src/views/auth/claims.ts" beforeDir="false" />
|
||||||
|
<change beforePath="$PROJECT_DIR$/web/src/views/auth/sign-in.vue" beforeDir="false" />
|
||||||
|
<change beforePath="$PROJECT_DIR$/web/src/views/auth/sign-up.vue" beforeDir="false" />
|
||||||
|
<change beforePath="$PROJECT_DIR$/web/src/views/dashboard.vue" beforeDir="false" />
|
||||||
|
<change beforePath="$PROJECT_DIR$/web/src/views/flow/confirm.vue" beforeDir="false" />
|
||||||
|
<change beforePath="$PROJECT_DIR$/web/src/views/flow/password-reset.vue" beforeDir="false" />
|
||||||
|
<change beforePath="$PROJECT_DIR$/web/src/views/personalize.vue" beforeDir="false" />
|
||||||
|
<change beforePath="$PROJECT_DIR$/web/src/views/security.vue" beforeDir="false" />
|
||||||
|
<change beforePath="$PROJECT_DIR$/web/tsconfig.app.json" beforeDir="false" />
|
||||||
|
<change beforePath="$PROJECT_DIR$/web/tsconfig.json" beforeDir="false" />
|
||||||
|
<change beforePath="$PROJECT_DIR$/web/tsconfig.node.json" beforeDir="false" />
|
||||||
|
<change beforePath="$PROJECT_DIR$/web/uno.config.ts" beforeDir="false" />
|
||||||
|
<change beforePath="$PROJECT_DIR$/web/vite.config.ts" beforeDir="false" />
|
||||||
|
</list>
|
||||||
<option name="SHOW_DIALOG" value="false" />
|
<option name="SHOW_DIALOG" value="false" />
|
||||||
<option name="HIGHLIGHT_CONFLICTS" value="true" />
|
<option name="HIGHLIGHT_CONFLICTS" value="true" />
|
||||||
<option name="HIGHLIGHT_NON_ACTIVE_CHANGELIST" value="false" />
|
<option name="HIGHLIGHT_NON_ACTIVE_CHANGELIST" value="false" />
|
||||||
@ -38,34 +97,34 @@
|
|||||||
<option name="hideEmptyMiddlePackages" value="true" />
|
<option name="hideEmptyMiddlePackages" value="true" />
|
||||||
<option name="showLibraryContents" value="true" />
|
<option name="showLibraryContents" value="true" />
|
||||||
</component>
|
</component>
|
||||||
<component name="PropertiesComponent">{
|
<component name="PropertiesComponent"><![CDATA[{
|
||||||
"keyToString": {
|
"keyToString": {
|
||||||
"DefaultGoTemplateProperty": "Go File",
|
"DefaultGoTemplateProperty": "Go File",
|
||||||
"Go Build.Backend.executor": "Run",
|
"Go Build.Backend.executor": "Run",
|
||||||
"Go 构建.Backend.executor": "Run",
|
"Go 构建.Backend.executor": "Run",
|
||||||
"RunOnceActivity.ShowReadmeOnStart": "true",
|
"RunOnceActivity.ShowReadmeOnStart": "true",
|
||||||
"RunOnceActivity.go.formatter.settings.were.checked": "true",
|
"RunOnceActivity.go.formatter.settings.were.checked": "true",
|
||||||
"RunOnceActivity.go.migrated.go.modules.settings": "true",
|
"RunOnceActivity.go.migrated.go.modules.settings": "true",
|
||||||
"RunOnceActivity.go.modules.automatic.dependencies.download": "true",
|
"RunOnceActivity.go.modules.automatic.dependencies.download": "true",
|
||||||
"RunOnceActivity.go.modules.go.list.on.any.changes.was.set": "true",
|
"RunOnceActivity.go.modules.go.list.on.any.changes.was.set": "true",
|
||||||
"git-widget-placeholder": "master",
|
"git-widget-placeholder": "refactor/use-capital-as-frontend",
|
||||||
"go.import.settings.migrated": "true",
|
"go.import.settings.migrated": "true",
|
||||||
"go.sdk.automatically.set": "true",
|
"go.sdk.automatically.set": "true",
|
||||||
"last_opened_file_path": "/Users/littlesheep/Documents/Projects/Hydrogen/Passport/web/src/components/admin",
|
"last_opened_file_path": "/Users/littlesheep/Documents/Projects/Hydrogen/Passport/web/src/components/admin",
|
||||||
"node.js.detected.package.eslint": "true",
|
"node.js.detected.package.eslint": "true",
|
||||||
"node.js.selected.package.eslint": "(autodetect)",
|
"node.js.selected.package.eslint": "(autodetect)",
|
||||||
"nodejs_package_manager_path": "npm",
|
"nodejs_package_manager_path": "npm",
|
||||||
"run.code.analysis.last.selected.profile": "pProject Default",
|
"run.code.analysis.last.selected.profile": "pProject Default",
|
||||||
"settings.editor.selected.configurable": "preferences.pluginManager",
|
"settings.editor.selected.configurable": "preferences.pluginManager",
|
||||||
"ts.external.directory.path": "/Users/littlesheep/Documents/Projects/Hydrogen/Passport/web/node_modules/typescript/lib",
|
"ts.external.directory.path": "/Users/littlesheep/Documents/Projects/Hydrogen/Passport/web/node_modules/typescript/lib",
|
||||||
"vue.rearranger.settings.migration": "true"
|
"vue.rearranger.settings.migration": "true"
|
||||||
},
|
},
|
||||||
"keyToStringList": {
|
"keyToStringList": {
|
||||||
"DatabaseDriversLRU": [
|
"DatabaseDriversLRU": [
|
||||||
"postgresql"
|
"postgresql"
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
}</component>
|
}]]></component>
|
||||||
<component name="RecentsManager">
|
<component name="RecentsManager">
|
||||||
<key name="CopyFile.RECENT_KEYS">
|
<key name="CopyFile.RECENT_KEYS">
|
||||||
<recent name="$PROJECT_DIR$/web/src/components/admin" />
|
<recent name="$PROJECT_DIR$/web/src/components/admin" />
|
||||||
@ -109,8 +168,8 @@
|
|||||||
<component name="SharedIndexes">
|
<component name="SharedIndexes">
|
||||||
<attachedChunks>
|
<attachedChunks>
|
||||||
<set>
|
<set>
|
||||||
<option value="bundled-gosdk-33c477a475b1-e0158606a674-org.jetbrains.plugins.go.sharedIndexes.bundled-GO-241.18034.61" />
|
<option value="bundled-gosdk-5df93f7ad4aa-dfc284eb1eb8-org.jetbrains.plugins.go.sharedIndexes.bundled-GO-242.20224.306" />
|
||||||
<option value="bundled-js-predefined-1d06a55b98c1-0b3e54e931b4-JavaScript-GO-241.18034.61" />
|
<option value="bundled-js-predefined-d6986cc7102b-410509235cf1-JavaScript-GO-242.20224.306" />
|
||||||
</set>
|
</set>
|
||||||
</attachedChunks>
|
</attachedChunks>
|
||||||
</component>
|
</component>
|
||||||
|
@ -1,23 +1,14 @@
|
|||||||
# Building Backend
|
# Building Backend
|
||||||
FROM golang:alpine as passport-server
|
FROM golang:alpine as passport-server
|
||||||
|
|
||||||
RUN apk add nodejs npm
|
|
||||||
|
|
||||||
WORKDIR /source
|
WORKDIR /source
|
||||||
COPY . .
|
COPY . .
|
||||||
|
|
||||||
WORKDIR /source/web
|
|
||||||
RUN npm install
|
|
||||||
RUN npm run build
|
|
||||||
|
|
||||||
WORKDIR /source
|
|
||||||
RUN CGO_ENABLED=0 GOOS=linux GOARCH=amd64 go build -buildvcs -o /dist ./pkg/main.go
|
RUN CGO_ENABLED=0 GOOS=linux GOARCH=amd64 go build -buildvcs -o /dist ./pkg/main.go
|
||||||
|
|
||||||
# Runtime
|
# Runtime
|
||||||
FROM golang:alpine
|
FROM golang:alpine
|
||||||
|
|
||||||
COPY --from=passport-server /dist /passport/server
|
COPY --from=passport-server /dist /passport/server
|
||||||
COPY --from=passport-server /source/web/dist /passport/web
|
|
||||||
|
|
||||||
EXPOSE 8444
|
EXPOSE 8444
|
||||||
|
|
||||||
|
2
go.mod
2
go.mod
@ -5,7 +5,7 @@ go 1.21.6
|
|||||||
toolchain go1.22.1
|
toolchain go1.22.1
|
||||||
|
|
||||||
require (
|
require (
|
||||||
git.solsynth.dev/hydrogen/dealer v0.0.0-20240721055146-d74cdddbaf49
|
git.solsynth.dev/hydrogen/dealer v0.0.0-20240801060523-8cf0feb09a27
|
||||||
github.com/go-playground/validator/v10 v10.17.0
|
github.com/go-playground/validator/v10 v10.17.0
|
||||||
github.com/gofiber/fiber/v2 v2.52.4
|
github.com/gofiber/fiber/v2 v2.52.4
|
||||||
github.com/golang-jwt/jwt/v5 v5.2.0
|
github.com/golang-jwt/jwt/v5 v5.2.0
|
||||||
|
2
go.sum
2
go.sum
@ -1,5 +1,7 @@
|
|||||||
git.solsynth.dev/hydrogen/dealer v0.0.0-20240721055146-d74cdddbaf49 h1:DMmCBcnCO0qcER/p4EQ04CmWleb4YI3Br6QK5F8Q628=
|
git.solsynth.dev/hydrogen/dealer v0.0.0-20240721055146-d74cdddbaf49 h1:DMmCBcnCO0qcER/p4EQ04CmWleb4YI3Br6QK5F8Q628=
|
||||||
git.solsynth.dev/hydrogen/dealer v0.0.0-20240721055146-d74cdddbaf49/go.mod h1:IZd94qZZIj+MO9EqjGDqnAD9nWurlNPyhVPKemAY5lw=
|
git.solsynth.dev/hydrogen/dealer v0.0.0-20240721055146-d74cdddbaf49/go.mod h1:IZd94qZZIj+MO9EqjGDqnAD9nWurlNPyhVPKemAY5lw=
|
||||||
|
git.solsynth.dev/hydrogen/dealer v0.0.0-20240801060523-8cf0feb09a27 h1:KQzeOI2ou240SXiL1hxMYDvZpYKtCFblCGDusFyGyBY=
|
||||||
|
git.solsynth.dev/hydrogen/dealer v0.0.0-20240801060523-8cf0feb09a27/go.mod h1:IZd94qZZIj+MO9EqjGDqnAD9nWurlNPyhVPKemAY5lw=
|
||||||
github.com/BurntSushi/toml v1.3.2 h1:o7IhLm0Msx3BaB+n3Ag7L8EVlByGnpq14C4YWiu/gL8=
|
github.com/BurntSushi/toml v1.3.2 h1:o7IhLm0Msx3BaB+n3Ag7L8EVlByGnpq14C4YWiu/gL8=
|
||||||
github.com/BurntSushi/toml v1.3.2/go.mod h1:CxXYINrC8qIiEnFrOxCa7Jy5BFHlXnUU2pbicEuybxQ=
|
github.com/BurntSushi/toml v1.3.2/go.mod h1:CxXYINrC8qIiEnFrOxCa7Jy5BFHlXnUU2pbicEuybxQ=
|
||||||
github.com/DataDog/datadog-go v3.2.0+incompatible/go.mod h1:LButxg5PwREeZtORoXG3tL4fMGNddJ+vMq1mwgfaqoQ=
|
github.com/DataDog/datadog-go v3.2.0+incompatible/go.mod h1:LButxg5PwREeZtORoXG3tL4fMGNddJ+vMq1mwgfaqoQ=
|
||||||
|
@ -1,8 +1,6 @@
|
|||||||
id = "passport01"
|
id = "passport01"
|
||||||
name = "Solarpass"
|
name = "Solarpass"
|
||||||
|
|
||||||
frontend_app = "web/dist"
|
|
||||||
|
|
||||||
bind = "0.0.0.0:8444"
|
bind = "0.0.0.0:8444"
|
||||||
grpc_bind = "0.0.0.0:7444"
|
grpc_bind = "0.0.0.0:7444"
|
||||||
domain = "localhost"
|
domain = "localhost"
|
||||||
|
@ -1,18 +0,0 @@
|
|||||||
/* eslint-env node */
|
|
||||||
require("@rushstack/eslint-patch/modern-module-resolution")
|
|
||||||
|
|
||||||
module.exports = {
|
|
||||||
root: true,
|
|
||||||
extends: [
|
|
||||||
"plugin:vue/vue3-essential",
|
|
||||||
"eslint:recommended",
|
|
||||||
"@vue/eslint-config-typescript",
|
|
||||||
"@vue/eslint-config-prettier/skip-formatting",
|
|
||||||
],
|
|
||||||
parserOptions: {
|
|
||||||
ecmaVersion: "latest",
|
|
||||||
},
|
|
||||||
rules: {
|
|
||||||
"vue/multi-word-component-names": "off",
|
|
||||||
}
|
|
||||||
}
|
|
32
web/.gitignore
vendored
32
web/.gitignore
vendored
@ -1,32 +0,0 @@
|
|||||||
# Logs
|
|
||||||
logs
|
|
||||||
*.log
|
|
||||||
npm-debug.log*
|
|
||||||
yarn-debug.log*
|
|
||||||
yarn-error.log*
|
|
||||||
pnpm-debug.log*
|
|
||||||
lerna-debug.log*
|
|
||||||
|
|
||||||
node_modules
|
|
||||||
.DS_Store
|
|
||||||
dist
|
|
||||||
dist-ssr
|
|
||||||
coverage
|
|
||||||
*.local
|
|
||||||
|
|
||||||
/cypress/videos/
|
|
||||||
/cypress/screenshots/
|
|
||||||
|
|
||||||
# Editor directories and files
|
|
||||||
.vscode/*
|
|
||||||
!.vscode/extensions.json
|
|
||||||
.idea
|
|
||||||
*.suo
|
|
||||||
*.ntvs*
|
|
||||||
*.njsproj
|
|
||||||
*.sln
|
|
||||||
*.sw?
|
|
||||||
|
|
||||||
*.tsbuildinfo
|
|
||||||
|
|
||||||
.vite
|
|
@ -1,8 +0,0 @@
|
|||||||
{
|
|
||||||
"$schema": "https://json.schemastore.org/prettierrc",
|
|
||||||
"semi": false,
|
|
||||||
"tabWidth": 2,
|
|
||||||
"singleQuote": false,
|
|
||||||
"printWidth": 120,
|
|
||||||
"trailingComma": "all"
|
|
||||||
}
|
|
@ -1,85 +0,0 @@
|
|||||||
{
|
|
||||||
"hash": "8b6d0833",
|
|
||||||
"configHash": "9db2f33b",
|
|
||||||
"lockfileHash": "f7feab31",
|
|
||||||
"browserHash": "0decaeb8",
|
|
||||||
"optimized": {
|
|
||||||
"vue": {
|
|
||||||
"src": "../../node_modules/vue/dist/vue.runtime.esm-bundler.js",
|
|
||||||
"file": "vue.js",
|
|
||||||
"fileHash": "1913939b",
|
|
||||||
"needsInterop": false
|
|
||||||
},
|
|
||||||
"pinia": {
|
|
||||||
"src": "../../node_modules/pinia/dist/pinia.mjs",
|
|
||||||
"file": "pinia.js",
|
|
||||||
"fileHash": "4c101c90",
|
|
||||||
"needsInterop": false
|
|
||||||
},
|
|
||||||
"vuetify": {
|
|
||||||
"src": "../../node_modules/vuetify/lib/framework.mjs",
|
|
||||||
"file": "vuetify.js",
|
|
||||||
"fileHash": "3ab2542c",
|
|
||||||
"needsInterop": false
|
|
||||||
},
|
|
||||||
"vuetify/blueprints": {
|
|
||||||
"src": "../../node_modules/vuetify/lib/blueprints/index.mjs",
|
|
||||||
"file": "vuetify_blueprints.js",
|
|
||||||
"fileHash": "b34fe63c",
|
|
||||||
"needsInterop": false
|
|
||||||
},
|
|
||||||
"vuetify/components": {
|
|
||||||
"src": "../../node_modules/vuetify/lib/components/index.mjs",
|
|
||||||
"file": "vuetify_components.js",
|
|
||||||
"fileHash": "42f8a374",
|
|
||||||
"needsInterop": false
|
|
||||||
},
|
|
||||||
"vuetify/labs/components": {
|
|
||||||
"src": "../../node_modules/vuetify/lib/labs/components.mjs",
|
|
||||||
"file": "vuetify_labs_components.js",
|
|
||||||
"fileHash": "a46a672a",
|
|
||||||
"needsInterop": false
|
|
||||||
},
|
|
||||||
"vuetify/directives": {
|
|
||||||
"src": "../../node_modules/vuetify/lib/directives/index.mjs",
|
|
||||||
"file": "vuetify_directives.js",
|
|
||||||
"fileHash": "66c34130",
|
|
||||||
"needsInterop": false
|
|
||||||
},
|
|
||||||
"vue-router": {
|
|
||||||
"src": "../../node_modules/vue-router/dist/vue-router.mjs",
|
|
||||||
"file": "vue-router.js",
|
|
||||||
"fileHash": "56488480",
|
|
||||||
"needsInterop": false
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"chunks": {
|
|
||||||
"chunk-WKTZ3I3B": {
|
|
||||||
"file": "chunk-WKTZ3I3B.js"
|
|
||||||
},
|
|
||||||
"chunk-XJKCW2PU": {
|
|
||||||
"file": "chunk-XJKCW2PU.js"
|
|
||||||
},
|
|
||||||
"chunk-VFMM6PD2": {
|
|
||||||
"file": "chunk-VFMM6PD2.js"
|
|
||||||
},
|
|
||||||
"chunk-XREPMAG4": {
|
|
||||||
"file": "chunk-XREPMAG4.js"
|
|
||||||
},
|
|
||||||
"chunk-ZUZRGUJJ": {
|
|
||||||
"file": "chunk-ZUZRGUJJ.js"
|
|
||||||
},
|
|
||||||
"chunk-V6X3YB3T": {
|
|
||||||
"file": "chunk-V6X3YB3T.js"
|
|
||||||
},
|
|
||||||
"chunk-6CN2GOSH": {
|
|
||||||
"file": "chunk-6CN2GOSH.js"
|
|
||||||
},
|
|
||||||
"chunk-AYVSL3LM": {
|
|
||||||
"file": "chunk-AYVSL3LM.js"
|
|
||||||
},
|
|
||||||
"chunk-Q5PGHB6G": {
|
|
||||||
"file": "chunk-Q5PGHB6G.js"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,3 +0,0 @@
|
|||||||
{
|
|
||||||
"type": "module"
|
|
||||||
}
|
|
@ -1,39 +0,0 @@
|
|||||||
# views
|
|
||||||
|
|
||||||
This template should help get you started developing with Vue 3 in Vite.
|
|
||||||
|
|
||||||
## Recommended IDE Setup
|
|
||||||
|
|
||||||
[VSCode](https://code.visualstudio.com/) + [Volar](https://marketplace.visualstudio.com/items?itemName=Vue.volar) (and disable Vetur).
|
|
||||||
|
|
||||||
## Type Support for `.vue` Imports in TS
|
|
||||||
|
|
||||||
TypeScript cannot handle type information for `.vue` imports by default, so we replace the `tsc` CLI with `vue-tsc` for type checking. In editors, we need [Volar](https://marketplace.visualstudio.com/items?itemName=Vue.volar) to make the TypeScript language service aware of `.vue` types.
|
|
||||||
|
|
||||||
## Customize configuration
|
|
||||||
|
|
||||||
See [Vite Configuration Reference](https://vitejs.dev/config/).
|
|
||||||
|
|
||||||
## Project Setup
|
|
||||||
|
|
||||||
```sh
|
|
||||||
npm install
|
|
||||||
```
|
|
||||||
|
|
||||||
### Compile and Hot-Reload for Development
|
|
||||||
|
|
||||||
```sh
|
|
||||||
npm run dev
|
|
||||||
```
|
|
||||||
|
|
||||||
### Type-Check, Compile and Minify for Production
|
|
||||||
|
|
||||||
```sh
|
|
||||||
npm run build
|
|
||||||
```
|
|
||||||
|
|
||||||
### Lint with [ESLint](https://eslint.org/)
|
|
||||||
|
|
||||||
```sh
|
|
||||||
npm run lint
|
|
||||||
```
|
|
BIN
web/bun.lockb
BIN
web/bun.lockb
Binary file not shown.
1
web/env.d.ts
vendored
1
web/env.d.ts
vendored
@ -1 +0,0 @@
|
|||||||
/// <reference types="vite/client" />
|
|
@ -1,13 +0,0 @@
|
|||||||
<!doctype html>
|
|
||||||
<html lang="en">
|
|
||||||
<head>
|
|
||||||
<meta charset="UTF-8" />
|
|
||||||
<link rel="icon" type="image/xml+svg" href="/favicon.png" />
|
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
|
||||||
<title>Solarpass</title>
|
|
||||||
</head>
|
|
||||||
<body>
|
|
||||||
<div id="app"></div>
|
|
||||||
<script type="module" src="/src/main.ts"></script>
|
|
||||||
</body>
|
|
||||||
</html>
|
|
@ -1,45 +0,0 @@
|
|||||||
{
|
|
||||||
"name": "passport-web",
|
|
||||||
"version": "1.0.0",
|
|
||||||
"type": "module",
|
|
||||||
"scripts": {
|
|
||||||
"dev": "vite",
|
|
||||||
"build": "run-p type-check \"build-only {@}\" --",
|
|
||||||
"preview": "vite preview",
|
|
||||||
"build-only": "vite build",
|
|
||||||
"type-check": "vue-tsc --build --force",
|
|
||||||
"lint": "eslint . --ext .vue,.js,.jsx,.cjs,.mjs,.ts,.tsx,.cts,.mts --fix --ignore-path .gitignore",
|
|
||||||
"format": "prettier --write src/"
|
|
||||||
},
|
|
||||||
"dependencies": {
|
|
||||||
"@fontsource/roboto": "^5.0.13",
|
|
||||||
"@mdi/font": "^7.4.47",
|
|
||||||
"@unocss/reset": "^0.61.0",
|
|
||||||
"dompurify": "^3.1.5",
|
|
||||||
"marked": "^12.0.2",
|
|
||||||
"pinia": "^2.1.7",
|
|
||||||
"universal-cookie": "^7.1.4",
|
|
||||||
"unocss": "^0.61.0",
|
|
||||||
"vue": "^3.4.30",
|
|
||||||
"vue-router": "^4.4.0",
|
|
||||||
"vuetify": "^3.6.10"
|
|
||||||
},
|
|
||||||
"devDependencies": {
|
|
||||||
"@rushstack/eslint-patch": "^1.10.3",
|
|
||||||
"@tsconfig/node20": "^20.1.4",
|
|
||||||
"@types/dompurify": "^3.0.5",
|
|
||||||
"@types/node": "^20.14.8",
|
|
||||||
"@vitejs/plugin-vue": "^5.0.5",
|
|
||||||
"@vitejs/plugin-vue-jsx": "^3.1.0",
|
|
||||||
"@vue/eslint-config-prettier": "^8.0.0",
|
|
||||||
"@vue/eslint-config-typescript": "^12.0.0",
|
|
||||||
"@vue/tsconfig": "^0.5.1",
|
|
||||||
"eslint": "^8.57.0",
|
|
||||||
"eslint-plugin-vue": "^9.26.0",
|
|
||||||
"npm-run-all2": "^6.2.0",
|
|
||||||
"prettier": "^3.3.2",
|
|
||||||
"typescript": "^5.4.5",
|
|
||||||
"vite": "^5.3.1",
|
|
||||||
"vue-tsc": "^2.0.22"
|
|
||||||
}
|
|
||||||
}
|
|
Binary file not shown.
Before Width: | Height: | Size: 75 KiB |
@ -1,14 +0,0 @@
|
|||||||
html,
|
|
||||||
body,
|
|
||||||
#app,
|
|
||||||
.v-application {
|
|
||||||
font-family: "Roboto Sans", ui-sans-serif, system-ui, sans-serif;
|
|
||||||
}
|
|
||||||
|
|
||||||
.no-scrollbar {
|
|
||||||
scrollbar-width: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
.no-scrollbar::-webkit-scrollbar {
|
|
||||||
width: 0;
|
|
||||||
}
|
|
@ -1,6 +0,0 @@
|
|||||||
<template>
|
|
||||||
<div class="text-xs text-center opacity-80">
|
|
||||||
<p>Copyright © {{ new Date().getFullYear() }} Solsynth LLC</p>
|
|
||||||
<p>Powered by <a class="underline" href="https://git.solsynth.dev/Hydrogen/Passport">Hydrogen.Passport</a></p>
|
|
||||||
</div>
|
|
||||||
</template>
|
|
@ -1,19 +0,0 @@
|
|||||||
<template>
|
|
||||||
<v-card prepend-icon="mdi-cellphone-arrow-down-variant" title="Try the App">
|
|
||||||
<v-card-text>
|
|
||||||
<p>
|
|
||||||
Some features on Solarpass web was incomplete.
|
|
||||||
Go try out our brand-new all-in-one Solar Network application now!
|
|
||||||
</p>
|
|
||||||
</v-card-text>
|
|
||||||
<v-card-actions>
|
|
||||||
<v-btn prepend-icon="mdi-launch" href="https://lian.solsynth.dev" target="_blank">
|
|
||||||
Open in browser
|
|
||||||
</v-btn>
|
|
||||||
<v-btn prepend-icon="mdi-download" href="https://git.solsynth.dev/Hydrogen/Solian/releases" target="_blank"
|
|
||||||
color="teal">
|
|
||||||
Download now
|
|
||||||
</v-btn>
|
|
||||||
</v-card-actions>
|
|
||||||
</v-card>
|
|
||||||
</template>
|
|
@ -1,79 +0,0 @@
|
|||||||
<template>
|
|
||||||
<v-navigation-drawer
|
|
||||||
:model-value="props.open"
|
|
||||||
@update:model-value="(val: any) => emits('update:open', val)"
|
|
||||||
location="right"
|
|
||||||
temporary
|
|
||||||
order="0"
|
|
||||||
width="400"
|
|
||||||
>
|
|
||||||
<v-list-item prepend-icon="mdi-bell" title="Notifications" class="py-3"></v-list-item>
|
|
||||||
|
|
||||||
<v-divider color="black" class="mb-1" />
|
|
||||||
|
|
||||||
<v-list v-if="notify.notifications.length <= 0" density="compact">
|
|
||||||
<v-list-item
|
|
||||||
color="secondary"
|
|
||||||
prepend-icon="mdi-check"
|
|
||||||
title="All notifications read"
|
|
||||||
subtitle="There is no more new things for you..."
|
|
||||||
/>
|
|
||||||
</v-list>
|
|
||||||
|
|
||||||
<v-list v-else density="compact" lines="three">
|
|
||||||
<v-list-item v-for="(item, idx) in notify.notifications" :key="idx">
|
|
||||||
<template #title>{{ item.title }}</template>
|
|
||||||
<template #subtitle>{{ item.subtitle }}<br v-if="item.subtitle" />{{ item.body }}</template>
|
|
||||||
|
|
||||||
<template #append>
|
|
||||||
<v-btn icon="mdi-check" size="x-small" variant="text" :disabled="loading" @click="markAsRead(item, idx)" />
|
|
||||||
</template>
|
|
||||||
|
|
||||||
<div class="flex text-xs gap-1">
|
|
||||||
<a v-for="(link, idx) in item.links" :key="idx" class="mt-1 underline" target="_blank" :href="link.url">{{
|
|
||||||
link.label
|
|
||||||
}}</a>
|
|
||||||
</div>
|
|
||||||
</v-list-item>
|
|
||||||
</v-list>
|
|
||||||
</v-navigation-drawer>
|
|
||||||
|
|
||||||
<!-- @vue-ignore -->
|
|
||||||
<v-snackbar v-model="error" :timeout="5000">Something went wrong... {{ error }}</v-snackbar>
|
|
||||||
</template>
|
|
||||||
|
|
||||||
<script setup lang="ts">
|
|
||||||
import { request } from "@/scripts/request"
|
|
||||||
import { getAtk } from "@/stores/userinfo"
|
|
||||||
import { computed, onMounted, onUnmounted, ref } from "vue"
|
|
||||||
import { useNotifications } from "@/stores/notifications"
|
|
||||||
|
|
||||||
const props = defineProps<{ open: boolean }>()
|
|
||||||
const emits = defineEmits(["update:open"])
|
|
||||||
|
|
||||||
const notify = useNotifications()
|
|
||||||
|
|
||||||
const error = ref<string | null>(null)
|
|
||||||
const submitting = ref(false)
|
|
||||||
const loading = computed(() => notify.loading || submitting.value)
|
|
||||||
|
|
||||||
async function markAsRead(item: any, idx: number) {
|
|
||||||
submitting.value = true
|
|
||||||
const res = await request(`/api/notifications/${item.id}/read`, {
|
|
||||||
method: "PUT",
|
|
||||||
headers: { Authorization: `Bearer ${getAtk()}` },
|
|
||||||
})
|
|
||||||
if (res.status !== 200) {
|
|
||||||
error.value = await res.text()
|
|
||||||
} else {
|
|
||||||
notify.remove(idx)
|
|
||||||
error.value = null
|
|
||||||
}
|
|
||||||
submitting.value = false
|
|
||||||
}
|
|
||||||
|
|
||||||
notify.list()
|
|
||||||
|
|
||||||
onMounted(() => notify.connect())
|
|
||||||
onUnmounted(() => notify.disconnect())
|
|
||||||
</script>
|
|
@ -1,53 +0,0 @@
|
|||||||
<template>
|
|
||||||
<v-menu>
|
|
||||||
<template #activator="{ props }">
|
|
||||||
<v-btn flat exact v-bind="props" icon>
|
|
||||||
<v-avatar color="transparent" icon="mdi-account-circle" image="/api/users/me/avatar" />
|
|
||||||
</v-btn>
|
|
||||||
</template>
|
|
||||||
|
|
||||||
<v-list density="compact" v-if="!id.userinfo.isLoggedIn">
|
|
||||||
<v-list-item title="Sign in" prepend-icon="mdi-login-variant" :to="{ name: 'auth.sign-in' }" />
|
|
||||||
<v-list-item title="Create account" prepend-icon="mdi-account-plus" :to="{ name: 'auth.sign-up' }" />
|
|
||||||
</v-list>
|
|
||||||
<v-list density="compact" v-else>
|
|
||||||
<v-list-item :title="nickname" :subtitle="username" />
|
|
||||||
|
|
||||||
<v-divider class="border-opacity-50 my-2" />
|
|
||||||
|
|
||||||
<v-list-item title="Dashboard" prepend-icon="mdi-account-supervisor" exact :to="{ name: 'dashboard' }" />
|
|
||||||
<v-list-item title="Sign out" prepend-icon="mdi-logout" @click="signout"></v-list-item>
|
|
||||||
</v-list>
|
|
||||||
</v-menu>
|
|
||||||
</template>
|
|
||||||
|
|
||||||
<script setup lang="ts">
|
|
||||||
import { defaultUserinfo, useUserinfo } from "@/stores/userinfo"
|
|
||||||
import { computed } from "vue"
|
|
||||||
import Cookie from "universal-cookie"
|
|
||||||
|
|
||||||
const id = useUserinfo()
|
|
||||||
|
|
||||||
const username = computed(() => {
|
|
||||||
if (id.userinfo.isLoggedIn) {
|
|
||||||
return "@" + id.userinfo.data?.name
|
|
||||||
} else {
|
|
||||||
return "@visitor"
|
|
||||||
}
|
|
||||||
})
|
|
||||||
const nickname = computed(() => {
|
|
||||||
if (id.userinfo.isLoggedIn) {
|
|
||||||
return id.userinfo.data?.nick
|
|
||||||
} else {
|
|
||||||
return "Anonymous"
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
function signout() {
|
|
||||||
const ck = new Cookie();
|
|
||||||
ck.remove("__hydrogen_atk");
|
|
||||||
ck.remove("__hydrogen_rtk")
|
|
||||||
id.userinfo = defaultUserinfo
|
|
||||||
window.location.reload()
|
|
||||||
}
|
|
||||||
</script>
|
|
@ -1,165 +0,0 @@
|
|||||||
<template>
|
|
||||||
<v-dialog class="max-w-[720px]" :model-value="props.data != null"
|
|
||||||
@update:model-value="(val: boolean) => !val && emits('close')">
|
|
||||||
<template v-slot:default="{ isActive }">
|
|
||||||
<v-card title="Assign permissions" :subtitle="`To user @${props.data?.name}`" :loading="submitting">
|
|
||||||
<v-card-text>
|
|
||||||
<v-sheet elevation="2" rounded="lg">
|
|
||||||
<v-table density="comfortable">
|
|
||||||
<thead>
|
|
||||||
<tr>
|
|
||||||
<th class="text-left">
|
|
||||||
Key
|
|
||||||
</th>
|
|
||||||
<th class="text-left">
|
|
||||||
Value
|
|
||||||
</th>
|
|
||||||
</tr>
|
|
||||||
</thead>
|
|
||||||
<tbody>
|
|
||||||
<tr
|
|
||||||
v-for="[key, val] in Object.entries(perms)"
|
|
||||||
:key="key"
|
|
||||||
>
|
|
||||||
<td class="w-1/2">
|
|
||||||
<div>
|
|
||||||
<p>{{ key }}</p>
|
|
||||||
<div class="flex mx-[-8px]">
|
|
||||||
<v-btn color="error" text="Delete" variant="plain" size="x-small"
|
|
||||||
@click="() => deleteNode(key)" />
|
|
||||||
<v-btn class="ms-[-8px]" color="info" text="Change" variant="plain" size="x-small"
|
|
||||||
@click="() => changeNodeType(key)" />
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</td>
|
|
||||||
<td class="w-1/2">
|
|
||||||
<div class="w-full flex items-center">
|
|
||||||
<v-checkbox v-if="typeof val === 'boolean'" class="my-1" density="comfortable"
|
|
||||||
:hide-details="true"
|
|
||||||
v-model="perms[key]" />
|
|
||||||
<v-number-input v-else-if="typeof val === 'number'"
|
|
||||||
controlVariant="default"
|
|
||||||
:reverse="false"
|
|
||||||
:hideInput="false"
|
|
||||||
:inset="false"
|
|
||||||
class="font-mono my-2"
|
|
||||||
density="compact" :hide-details="true"
|
|
||||||
v-model="perms[key]" />
|
|
||||||
<v-text-field v-else class="font-mono my-2" density="compact" :hide-details="true"
|
|
||||||
v-model="perms[key]" />
|
|
||||||
</div>
|
|
||||||
</td>
|
|
||||||
</tr>
|
|
||||||
<tr>
|
|
||||||
<td>
|
|
||||||
<v-text-field class="my-3.5" label="Key" density="compact" variant="solo-filled"
|
|
||||||
v-model="pendingNodeKey"
|
|
||||||
:hide-details="true" />
|
|
||||||
</td>
|
|
||||||
<td>
|
|
||||||
<div class="w-full flex justify-center">
|
|
||||||
<v-btn prepend-icon="mdi-plus-circle" text="Add one" block rounded="md" @click="addNode" />
|
|
||||||
</div>
|
|
||||||
</td>
|
|
||||||
</tr>
|
|
||||||
</tbody>
|
|
||||||
</v-table>
|
|
||||||
</v-sheet>
|
|
||||||
</v-card-text>
|
|
||||||
|
|
||||||
<v-card-actions>
|
|
||||||
<v-spacer></v-spacer>
|
|
||||||
|
|
||||||
<v-btn
|
|
||||||
:disabled="submitting"
|
|
||||||
text="Cancel"
|
|
||||||
color="grey"
|
|
||||||
@click="isActive.value = false"
|
|
||||||
></v-btn>
|
|
||||||
<v-btn
|
|
||||||
:disabled="submitting"
|
|
||||||
text="Apply Changes"
|
|
||||||
@click="saveNode"
|
|
||||||
></v-btn>
|
|
||||||
</v-card-actions>
|
|
||||||
</v-card>
|
|
||||||
</template>
|
|
||||||
</v-dialog>
|
|
||||||
</template>
|
|
||||||
|
|
||||||
<script setup lang="ts">
|
|
||||||
import { ref, watch } from "vue"
|
|
||||||
import { request } from "@/scripts/request"
|
|
||||||
import { getAtk } from "@/stores/userinfo"
|
|
||||||
|
|
||||||
const perms = ref<any>({})
|
|
||||||
|
|
||||||
const pendingNodeKey = ref("")
|
|
||||||
|
|
||||||
const props = defineProps<{ data: any }>()
|
|
||||||
const emits = defineEmits(["close", "success", "error"])
|
|
||||||
|
|
||||||
watch(props, (v) => {
|
|
||||||
if (v.data != null) {
|
|
||||||
perms.value = v.data["perm_nodes"]
|
|
||||||
}
|
|
||||||
}, { immediate: true, deep: true })
|
|
||||||
|
|
||||||
function addNode() {
|
|
||||||
if (pendingNodeKey.value) {
|
|
||||||
perms.value[pendingNodeKey.value] = false
|
|
||||||
pendingNodeKey.value = ""
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function deleteNode(key: string) {
|
|
||||||
delete perms.value[key]
|
|
||||||
}
|
|
||||||
|
|
||||||
function changeNodeType(key: string) {
|
|
||||||
const typelist = [
|
|
||||||
"boolean",
|
|
||||||
"number",
|
|
||||||
"string",
|
|
||||||
]
|
|
||||||
const idx = typelist.indexOf(typeof perms.value[key])
|
|
||||||
if (idx == -1 || idx == typelist.length - 1) {
|
|
||||||
perms.value[key] = false
|
|
||||||
return
|
|
||||||
}
|
|
||||||
switch (typelist[idx + 1]) {
|
|
||||||
case "boolean":
|
|
||||||
perms.value[key] = false
|
|
||||||
break
|
|
||||||
case "number":
|
|
||||||
perms.value[key] = 0
|
|
||||||
break
|
|
||||||
default:
|
|
||||||
perms.value[key] = ""
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const submitting = ref(false)
|
|
||||||
|
|
||||||
async function saveNode() {
|
|
||||||
submitting.value = true
|
|
||||||
const res = await request(`/api/admin/users/${props.data.id}/permissions`, {
|
|
||||||
method: "PUT",
|
|
||||||
headers: {
|
|
||||||
"Content-Type": "application/json",
|
|
||||||
"Authorization": `Bearer ${getAtk()}`,
|
|
||||||
},
|
|
||||||
body: JSON.stringify({
|
|
||||||
"perm_nodes": perms.value,
|
|
||||||
}),
|
|
||||||
})
|
|
||||||
if (res.status !== 200) {
|
|
||||||
emits("error", await res.text())
|
|
||||||
} else {
|
|
||||||
emits("success")
|
|
||||||
emits("close")
|
|
||||||
}
|
|
||||||
submitting.value = false
|
|
||||||
}
|
|
||||||
</script>
|
|
@ -1,47 +0,0 @@
|
|||||||
<template>
|
|
||||||
<v-dialog class="max-w-[720px]" :model-value="props.data != null"
|
|
||||||
@update:model-value="(val: boolean) => !val && emits('close')">
|
|
||||||
<template v-slot:default="{ isActive }">
|
|
||||||
<v-card :title="`User @${props.data?.name}`">
|
|
||||||
<v-card-text>
|
|
||||||
<v-row>
|
|
||||||
<v-col cols="12" md="6">
|
|
||||||
<h4 class="field-title">Name</h4>
|
|
||||||
<p>{{ props.data?.name }}</p>
|
|
||||||
</v-col>
|
|
||||||
<v-col cols="12" md="6">
|
|
||||||
<h4 class="field-title">Nick</h4>
|
|
||||||
<p>{{ props.data?.nick }}</p>
|
|
||||||
</v-col>
|
|
||||||
<v-col cols="12">
|
|
||||||
<h4 class="field-title">Entire Payload</h4>
|
|
||||||
<v-code class="font-mono overflow-x-scroll max-h-[360px]">
|
|
||||||
<pre>{{ JSON.stringify(props.data, null, 4) }}</pre>
|
|
||||||
</v-code>
|
|
||||||
</v-col>
|
|
||||||
</v-row>
|
|
||||||
</v-card-text>
|
|
||||||
|
|
||||||
<v-card-actions>
|
|
||||||
<v-spacer></v-spacer>
|
|
||||||
|
|
||||||
<v-btn
|
|
||||||
text="Close"
|
|
||||||
@click="isActive.value = false"
|
|
||||||
></v-btn>
|
|
||||||
</v-card-actions>
|
|
||||||
</v-card>
|
|
||||||
</template>
|
|
||||||
</v-dialog>
|
|
||||||
</template>
|
|
||||||
|
|
||||||
<script setup lang="ts">
|
|
||||||
const props = defineProps<{ data: any }>()
|
|
||||||
const emits = defineEmits(["close"])
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<style scoped>
|
|
||||||
.field-title {
|
|
||||||
font-weight: bold;
|
|
||||||
}
|
|
||||||
</style>
|
|
@ -1,74 +0,0 @@
|
|||||||
<template>
|
|
||||||
<v-dialog class="max-w-[720px]" :model-value="props.data != null"
|
|
||||||
@update:model-value="(val: boolean) => !val && emits('close')"
|
|
||||||
:loading="reverting">
|
|
||||||
<template v-slot:default="{ isActive }">
|
|
||||||
<v-card title="Auth Factors" :subtitle="`Of user @${props.data?.name}`">
|
|
||||||
<v-card-text>
|
|
||||||
<v-sheet elevation="2" rounded="lg">
|
|
||||||
<v-table density="compact">
|
|
||||||
<thead>
|
|
||||||
<tr>
|
|
||||||
<th class="text-left">
|
|
||||||
Name
|
|
||||||
</th>
|
|
||||||
<th class="text-left">
|
|
||||||
Secret
|
|
||||||
</th>
|
|
||||||
</tr>
|
|
||||||
</thead>
|
|
||||||
<tbody>
|
|
||||||
<tr
|
|
||||||
v-for="item in factors"
|
|
||||||
:key="item.name"
|
|
||||||
>
|
|
||||||
<td class="w-1/2">{{ item.id }}</td>
|
|
||||||
<td class="w-1/2"><code>{{ item.secret }}</code></td>
|
|
||||||
</tr>
|
|
||||||
</tbody>
|
|
||||||
</v-table>
|
|
||||||
</v-sheet>
|
|
||||||
</v-card-text>
|
|
||||||
|
|
||||||
<v-card-actions>
|
|
||||||
<v-spacer></v-spacer>
|
|
||||||
|
|
||||||
<v-btn
|
|
||||||
text="Close"
|
|
||||||
@click="isActive.value = false"
|
|
||||||
></v-btn>
|
|
||||||
</v-card-actions>
|
|
||||||
</v-card>
|
|
||||||
</template>
|
|
||||||
</v-dialog>
|
|
||||||
</template>
|
|
||||||
|
|
||||||
<script setup lang="ts">
|
|
||||||
import { ref, watch } from "vue"
|
|
||||||
import { request } from "@/scripts/request"
|
|
||||||
|
|
||||||
const props = defineProps<{ data: any }>()
|
|
||||||
const emits = defineEmits(["close", "error"])
|
|
||||||
|
|
||||||
const reverting = ref(false)
|
|
||||||
|
|
||||||
const factors = ref<any[]>([])
|
|
||||||
|
|
||||||
async function load() {
|
|
||||||
reverting.value = true
|
|
||||||
const res = await request(`/api/admin/users/${props.data.id}/factors`)
|
|
||||||
if (res.status !== 200) {
|
|
||||||
emits("error", await res.text())
|
|
||||||
} else {
|
|
||||||
factors.value = await res.json()
|
|
||||||
}
|
|
||||||
reverting.value = false
|
|
||||||
}
|
|
||||||
|
|
||||||
watch(props, (v) => {
|
|
||||||
if (v.data != null) {
|
|
||||||
factors.value = []
|
|
||||||
load()
|
|
||||||
}
|
|
||||||
}, { immediate: true, deep: true })
|
|
||||||
</script>
|
|
@ -1,65 +0,0 @@
|
|||||||
<template>
|
|
||||||
<div class="flex items-center">
|
|
||||||
<v-form class="flex-grow-1" @submit.prevent="submit">
|
|
||||||
<v-text-field label="Username" variant="solo" density="comfortable" class="mb-3" :hide-details="true"
|
|
||||||
:disabled="props.loading" v-model="probe" />
|
|
||||||
<v-text-field label="Password" variant="solo" density="comfortable" type="password" :disabled="props.loading"
|
|
||||||
v-model="password" />
|
|
||||||
|
|
||||||
<v-expand-transition>
|
|
||||||
<v-alert v-show="error" variant="tonal" type="error" class="text-xs mb-3">
|
|
||||||
Something went wrong... {{ error }}
|
|
||||||
</v-alert>
|
|
||||||
</v-expand-transition>
|
|
||||||
|
|
||||||
<div class="flex justify-between">
|
|
||||||
<v-btn type="button" variant="plain" color="grey-darken-3" :to="{ name: 'auth.sign-up' }">Sign up</v-btn>
|
|
||||||
|
|
||||||
<v-btn
|
|
||||||
type="submit"
|
|
||||||
variant="text"
|
|
||||||
color="primary"
|
|
||||||
class="justify-self-end"
|
|
||||||
append-icon="mdi-arrow-right"
|
|
||||||
:disabled="props.loading"
|
|
||||||
>
|
|
||||||
Next
|
|
||||||
</v-btn>
|
|
||||||
</div>
|
|
||||||
</v-form>
|
|
||||||
</div>
|
|
||||||
</template>
|
|
||||||
|
|
||||||
<script setup lang="ts">
|
|
||||||
import { ref } from "vue"
|
|
||||||
import { request } from "@/scripts/request"
|
|
||||||
|
|
||||||
const probe = ref("")
|
|
||||||
const password = ref("")
|
|
||||||
|
|
||||||
const error = ref<string | null>(null)
|
|
||||||
|
|
||||||
const props = defineProps<{ loading?: boolean }>()
|
|
||||||
const emits = defineEmits(["swap", "update:loading", "update:ticket"])
|
|
||||||
|
|
||||||
async function submit() {
|
|
||||||
if (!probe.value || !password.value) return
|
|
||||||
|
|
||||||
emits("update:loading", true)
|
|
||||||
const res = await request("/api/auth", {
|
|
||||||
method: "POST",
|
|
||||||
headers: { "Content-Type": "application/json" },
|
|
||||||
body: JSON.stringify({ username: probe.value, password: password.value }),
|
|
||||||
})
|
|
||||||
if (res.status !== 200) {
|
|
||||||
error.value = await res.text()
|
|
||||||
} else {
|
|
||||||
const data = await res.json()
|
|
||||||
emits("update:ticket", data["ticket"])
|
|
||||||
if (data.is_finished) emits("swap", "completed")
|
|
||||||
else emits("swap", "mfa")
|
|
||||||
error.value = null
|
|
||||||
}
|
|
||||||
emits("update:loading", false)
|
|
||||||
}
|
|
||||||
</script>
|
|
@ -1,67 +0,0 @@
|
|||||||
<template>
|
|
||||||
<div>
|
|
||||||
<v-icon icon="mdi-lan-check" size="32" color="grey-darken-3" class="mb-3" />
|
|
||||||
|
|
||||||
<h1 class="font-bold text-xl">All Done!</h1>
|
|
||||||
<p>Welcome back! You just signed in right now! We're going to send you to jesus...</p>
|
|
||||||
|
|
||||||
<v-expand-transition>
|
|
||||||
<v-alert v-show="error" variant="tonal" type="error" class="text-xs mb-3">
|
|
||||||
Something went wrong... {{ error }}
|
|
||||||
</v-alert>
|
|
||||||
</v-expand-transition>
|
|
||||||
</div>
|
|
||||||
</template>
|
|
||||||
|
|
||||||
<script setup lang="ts">
|
|
||||||
import { request } from "@/scripts/request"
|
|
||||||
import { useUserinfo } from "@/stores/userinfo"
|
|
||||||
import { onMounted, ref } from "vue"
|
|
||||||
import { useRoute, useRouter } from "vue-router"
|
|
||||||
|
|
||||||
const route = useRoute()
|
|
||||||
const router = useRouter()
|
|
||||||
const userinfo = useUserinfo()
|
|
||||||
|
|
||||||
const props = defineProps<{ loading?: boolean; currentFactor?: any; ticket?: any }>()
|
|
||||||
const emits = defineEmits(["update:loading"])
|
|
||||||
|
|
||||||
const error = ref<string | null>(null)
|
|
||||||
|
|
||||||
async function load() {
|
|
||||||
emits("update:loading", true)
|
|
||||||
await getToken(props.ticket.grant_token)
|
|
||||||
await userinfo.readProfiles()
|
|
||||||
setTimeout(() => callback(), 1850)
|
|
||||||
}
|
|
||||||
|
|
||||||
onMounted(() => load())
|
|
||||||
|
|
||||||
async function getToken(tk: string) {
|
|
||||||
const res = await request("/api/auth/token", {
|
|
||||||
method: "POST",
|
|
||||||
headers: { "Content-Type": "application/json" },
|
|
||||||
body: JSON.stringify({
|
|
||||||
code: tk,
|
|
||||||
grant_type: "grant_token",
|
|
||||||
}),
|
|
||||||
})
|
|
||||||
if (res.status !== 200) {
|
|
||||||
const err = await res.text()
|
|
||||||
error.value = err
|
|
||||||
throw new Error(err)
|
|
||||||
} else {
|
|
||||||
error.value = null
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function callback() {
|
|
||||||
if (route.query["close"]) {
|
|
||||||
window.close()
|
|
||||||
} else if (route.query["redirect_uri"]) {
|
|
||||||
window.open((route.query["redirect_uri"] as string) ?? "/", "_self")
|
|
||||||
} else {
|
|
||||||
router.push({ name: "dashboard" })
|
|
||||||
}
|
|
||||||
}
|
|
||||||
</script>
|
|
@ -1,16 +0,0 @@
|
|||||||
<template>
|
|
||||||
<div class="w-full max-w-[720px]">
|
|
||||||
<v-expand-transition>
|
|
||||||
<v-alert v-show="route.query['redirect_uri']" variant="tonal" type="info" class="text-xs">
|
|
||||||
You need to sign in before access that page. After you signed in, we will redirect you to: <br />
|
|
||||||
<span class="font-mono">{{ route.query["redirect_uri"] }}</span>
|
|
||||||
</v-alert>
|
|
||||||
</v-expand-transition>
|
|
||||||
</div>
|
|
||||||
</template>
|
|
||||||
|
|
||||||
<script setup lang="ts">
|
|
||||||
import { useRoute } from "vue-router"
|
|
||||||
|
|
||||||
const route = useRoute()
|
|
||||||
</script>
|
|
@ -1,93 +0,0 @@
|
|||||||
<template>
|
|
||||||
<div class="flex items-center">
|
|
||||||
<v-form class="flex-grow-1" @submit.prevent="submit">
|
|
||||||
<div v-if="inputType === 'one-time-password'" class="text-center">
|
|
||||||
<p class="text-xs opacity-90">Check your inbox!</p>
|
|
||||||
<v-otp-input
|
|
||||||
class="pt-0"
|
|
||||||
variant="solo"
|
|
||||||
density="compact"
|
|
||||||
type="text"
|
|
||||||
:length="6"
|
|
||||||
v-model="password"
|
|
||||||
:loading="loading"
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
<v-text-field
|
|
||||||
v-else
|
|
||||||
label="Password"
|
|
||||||
type="password"
|
|
||||||
variant="solo"
|
|
||||||
density="comfortable"
|
|
||||||
:disabled="loading"
|
|
||||||
v-model="password"
|
|
||||||
/>
|
|
||||||
|
|
||||||
<v-expand-transition>
|
|
||||||
<v-alert v-show="error" variant="tonal" type="error" class="text-xs mb-3">
|
|
||||||
Something went wrong... {{ error }}
|
|
||||||
</v-alert>
|
|
||||||
</v-expand-transition>
|
|
||||||
|
|
||||||
<div class="flex justify-end">
|
|
||||||
<v-btn
|
|
||||||
type="submit"
|
|
||||||
variant="text"
|
|
||||||
color="primary"
|
|
||||||
class="justify-self-end"
|
|
||||||
append-icon="mdi-arrow-right"
|
|
||||||
:disabled="loading"
|
|
||||||
>
|
|
||||||
Next
|
|
||||||
</v-btn>
|
|
||||||
</div>
|
|
||||||
</v-form>
|
|
||||||
</div>
|
|
||||||
</template>
|
|
||||||
|
|
||||||
<script setup lang="ts">
|
|
||||||
import { request } from "@/scripts/request"
|
|
||||||
import { computed, ref } from "vue"
|
|
||||||
|
|
||||||
const password = ref("")
|
|
||||||
|
|
||||||
const error = ref<string | null>(null)
|
|
||||||
|
|
||||||
const props = defineProps<{ loading?: boolean; currentFactor?: any; ticket?: any }>()
|
|
||||||
const emits = defineEmits(["swap", "update:ticket", "update:loading"])
|
|
||||||
|
|
||||||
async function submit() {
|
|
||||||
emits("update:loading", true)
|
|
||||||
const res = await request(`/api/auth/mfa`, {
|
|
||||||
method: "POST",
|
|
||||||
headers: { "Content-Type": "application/json" },
|
|
||||||
body: JSON.stringify({
|
|
||||||
ticket_id: props.ticket?.id,
|
|
||||||
factor_id: props.currentFactor?.id,
|
|
||||||
code: password.value,
|
|
||||||
}),
|
|
||||||
})
|
|
||||||
if (res.status !== 200) {
|
|
||||||
error.value = await res.text()
|
|
||||||
} else {
|
|
||||||
const data = await res.json()
|
|
||||||
error.value = null
|
|
||||||
password.value = ""
|
|
||||||
emits("update:ticket", data["ticket"])
|
|
||||||
if (data["is_finished"]) emits("swap", "completed")
|
|
||||||
else emits("swap", "mfa")
|
|
||||||
}
|
|
||||||
emits("update:loading", false)
|
|
||||||
}
|
|
||||||
|
|
||||||
const inputType = computed(() => {
|
|
||||||
switch (props.currentFactor?.type) {
|
|
||||||
case 0:
|
|
||||||
return "text"
|
|
||||||
case 1:
|
|
||||||
return "one-time-password"
|
|
||||||
default:
|
|
||||||
return "unknown"
|
|
||||||
}
|
|
||||||
})
|
|
||||||
</script>
|
|
@ -1,88 +0,0 @@
|
|||||||
<template>
|
|
||||||
<div class="flex items-center">
|
|
||||||
<div class="flex-grow-1">
|
|
||||||
<v-card class="mb-3">
|
|
||||||
<v-list density="compact" color="primary">
|
|
||||||
<v-list-item
|
|
||||||
v-for="(item, idx) in factors ?? []"
|
|
||||||
:key="idx"
|
|
||||||
:prepend-icon="getFactorType(item)?.icon"
|
|
||||||
:title="getFactorType(item)?.label"
|
|
||||||
:active="focus === item.id"
|
|
||||||
:disabled="getFactorAvailable(item)"
|
|
||||||
@click="focus = item.id"
|
|
||||||
/>
|
|
||||||
</v-list>
|
|
||||||
</v-card>
|
|
||||||
|
|
||||||
<v-expand-transition>
|
|
||||||
<v-alert v-show="error" variant="tonal" type="error" class="text-xs mb-3">
|
|
||||||
Something went wrong... {{ error }}
|
|
||||||
</v-alert>
|
|
||||||
</v-expand-transition>
|
|
||||||
|
|
||||||
<div class="flex justify-end">
|
|
||||||
<v-btn variant="text" color="primary" class="justify-self-end" append-icon="mdi-arrow-right" @click="submit">
|
|
||||||
Next
|
|
||||||
</v-btn>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</template>
|
|
||||||
|
|
||||||
<script setup lang="ts">
|
|
||||||
import { onMounted, ref } from "vue"
|
|
||||||
import { request } from "@/scripts/request"
|
|
||||||
|
|
||||||
const focus = ref<number | null>(null)
|
|
||||||
const factors = ref<any[]>([])
|
|
||||||
|
|
||||||
const error = ref<string | null>(null)
|
|
||||||
|
|
||||||
const props = defineProps<{ ticket?: any }>()
|
|
||||||
const emits = defineEmits(["swap", "update:loading", "update:currentFactor"])
|
|
||||||
|
|
||||||
async function load() {
|
|
||||||
emits("update:loading", true)
|
|
||||||
const res = await request(`/api/auth/factors?ticketId=${props.ticket.id}`)
|
|
||||||
if (res.status !== 200) {
|
|
||||||
error.value = await res.text()
|
|
||||||
} else {
|
|
||||||
factors.value = (await res.json()).filter((e: any) => e.type != 0)
|
|
||||||
}
|
|
||||||
emits("update:loading", false)
|
|
||||||
}
|
|
||||||
|
|
||||||
onMounted(() => load())
|
|
||||||
|
|
||||||
async function submit() {
|
|
||||||
if (!focus.value) return
|
|
||||||
|
|
||||||
emits("update:loading", true)
|
|
||||||
const res = await request(`/api/auth/factors/${focus.value}`, {
|
|
||||||
method: "POST",
|
|
||||||
})
|
|
||||||
if (res.status !== 200 && res.status !== 204) {
|
|
||||||
error.value = await res.text()
|
|
||||||
} else {
|
|
||||||
const item = factors.value.find((item: any) => item.id === focus.value)
|
|
||||||
emits("update:currentFactor", item)
|
|
||||||
emits("swap", "applicator")
|
|
||||||
error.value = null
|
|
||||||
focus.value = null
|
|
||||||
}
|
|
||||||
emits("update:loading", false)
|
|
||||||
}
|
|
||||||
|
|
||||||
function getFactorType(item: any) {
|
|
||||||
switch (item.type) {
|
|
||||||
case 1:
|
|
||||||
return { icon: "mdi-email-fast", label: "Email Validation" }
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function getFactorAvailable(factor: any) {
|
|
||||||
const blacklist: number[] = props.ticket?.blacklist_factors ?? []
|
|
||||||
return blacklist.includes(factor.id)
|
|
||||||
}
|
|
||||||
</script>
|
|
@ -1,52 +0,0 @@
|
|||||||
<template>
|
|
||||||
<v-app-bar height="64" color="primary" scroll-behavior="elevate" flat>
|
|
||||||
<div class="max-md:px-5 md:px-12 flex flex-grow-1 items-center">
|
|
||||||
<router-link :to="{ name: 'dashboard' }" class="flex gap-1 ms-0.5">
|
|
||||||
<img src="/favicon.png" alt="logo" width="27" height="24" class="icon-filter" />
|
|
||||||
<h2 class="ml-2 text-lg font-500">{{ props.title ?? "Solarpass" }}</h2>
|
|
||||||
</router-link>
|
|
||||||
|
|
||||||
<v-spacer />
|
|
||||||
|
|
||||||
<div class="me-2">
|
|
||||||
<v-btn icon size="small" variant="text" @click="openNotify = !openNotify">
|
|
||||||
<v-badge v-if="notify.total > 0" color="error" :content="notify.total">
|
|
||||||
<v-icon icon="mdi-bell" />
|
|
||||||
</v-badge>
|
|
||||||
|
|
||||||
<v-icon v-else icon="mdi-bell" />
|
|
||||||
</v-btn>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div>
|
|
||||||
<user-menu />
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<template v-if="slots.extension" #extension>
|
|
||||||
<slot name="extension" />
|
|
||||||
</template>
|
|
||||||
</v-app-bar>
|
|
||||||
|
|
||||||
<NotificationList v-model:open="openNotify" />
|
|
||||||
</template>
|
|
||||||
|
|
||||||
<script setup lang="ts">
|
|
||||||
import NotificationList from "@/components/NotificationList.vue"
|
|
||||||
import UserMenu from "@/components/UserMenu.vue"
|
|
||||||
import { useNotifications } from "@/stores/notifications"
|
|
||||||
import { ref, useSlots } from "vue"
|
|
||||||
|
|
||||||
const notify = useNotifications()
|
|
||||||
|
|
||||||
const slots = useSlots()
|
|
||||||
const props = defineProps<{ title?: String }>()
|
|
||||||
|
|
||||||
const openNotify = ref(false)
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<style scoped>
|
|
||||||
.icon-filter {
|
|
||||||
filter: invert(100%) sepia(100%) saturate(14%) hue-rotate(212deg) brightness(104%) contrast(104%);
|
|
||||||
}
|
|
||||||
</style>
|
|
@ -1,5 +0,0 @@
|
|||||||
<template>
|
|
||||||
<v-app>
|
|
||||||
<router-view />
|
|
||||||
</v-app>
|
|
||||||
</template>
|
|
@ -1,30 +0,0 @@
|
|||||||
<template>
|
|
||||||
<app-bar title="Solarpass Administration" />
|
|
||||||
|
|
||||||
<v-main>
|
|
||||||
<router-view />
|
|
||||||
</v-main>
|
|
||||||
</template>
|
|
||||||
|
|
||||||
<script setup lang="ts">
|
|
||||||
import { useUserinfo } from "@/stores/userinfo"
|
|
||||||
import { useRouter } from "vue-router"
|
|
||||||
import { onMounted } from "vue"
|
|
||||||
import AppBar from "@/components/navigation/AppBar.vue"
|
|
||||||
|
|
||||||
const id = useUserinfo()
|
|
||||||
const router = useRouter()
|
|
||||||
|
|
||||||
onMounted(async () => {
|
|
||||||
await id.readProfiles()
|
|
||||||
if (!id.userinfo.data.perm_nodes["AdminView"]) {
|
|
||||||
await router.push({ name: "dashboard" })
|
|
||||||
}
|
|
||||||
})
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<style scoped>
|
|
||||||
.icon-filter {
|
|
||||||
filter: invert(100%) sepia(100%) saturate(14%) hue-rotate(212deg) brightness(104%) contrast(104%);
|
|
||||||
}
|
|
||||||
</style>
|
|
@ -1,22 +0,0 @@
|
|||||||
<template>
|
|
||||||
<app-bar />
|
|
||||||
|
|
||||||
<v-main>
|
|
||||||
<router-view />
|
|
||||||
</v-main>
|
|
||||||
</template>
|
|
||||||
|
|
||||||
<script setup lang="ts">
|
|
||||||
import { useUserinfo } from "@/stores/userinfo"
|
|
||||||
import AppBar from "@/components/navigation/AppBar.vue"
|
|
||||||
|
|
||||||
const id = useUserinfo()
|
|
||||||
|
|
||||||
id.readProfiles()
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<style scoped>
|
|
||||||
.icon-filter {
|
|
||||||
filter: invert(100%) sepia(100%) saturate(14%) hue-rotate(212deg) brightness(104%) contrast(104%);
|
|
||||||
}
|
|
||||||
</style>
|
|
@ -1,30 +0,0 @@
|
|||||||
<template>
|
|
||||||
<app-bar>
|
|
||||||
<template #extension>
|
|
||||||
<v-tabs align-tabs="title" color="white">
|
|
||||||
<v-tab text="Dashboard" prepend-icon="mdi-view-dashboard" :to="{ name: 'dashboard' }" exact />
|
|
||||||
<v-tab text="Personalize" prepend-icon="mdi-card-bulleted-outline" :to="{ name: 'personalize' }" exact />
|
|
||||||
<v-tab text="Security" prepend-icon="mdi-security" :to="{ name: 'security' }" exact />
|
|
||||||
</v-tabs>
|
|
||||||
</template>
|
|
||||||
</app-bar>
|
|
||||||
|
|
||||||
<v-main>
|
|
||||||
<v-container class="pt-6 px-6 p-container">
|
|
||||||
<router-view />
|
|
||||||
</v-container>
|
|
||||||
|
|
||||||
<Copyright />
|
|
||||||
</v-main>
|
|
||||||
</template>
|
|
||||||
|
|
||||||
<script setup lang="ts">
|
|
||||||
import AppBar from "@/components/navigation/AppBar.vue"
|
|
||||||
import Copyright from "@/components/Copyright.vue"
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<style scoped>
|
|
||||||
.p-container {
|
|
||||||
max-width: 64rem;
|
|
||||||
}
|
|
||||||
</style>
|
|
@ -1,54 +0,0 @@
|
|||||||
import "virtual:uno.css"
|
|
||||||
|
|
||||||
import "./assets/utils.css"
|
|
||||||
|
|
||||||
import { createApp } from "vue"
|
|
||||||
import { createPinia } from "pinia"
|
|
||||||
|
|
||||||
import "vuetify/styles"
|
|
||||||
import { createVuetify } from "vuetify"
|
|
||||||
import { md3 } from "vuetify/blueprints"
|
|
||||||
import * as components from "vuetify/components"
|
|
||||||
import * as labsComponents from "vuetify/labs/components"
|
|
||||||
import * as directives from "vuetify/directives"
|
|
||||||
|
|
||||||
import "@mdi/font/css/materialdesignicons.min.css"
|
|
||||||
import "@fontsource/roboto/latin.css"
|
|
||||||
import "@unocss/reset/tailwind.css"
|
|
||||||
|
|
||||||
import index from "./index.vue"
|
|
||||||
import router from "./router"
|
|
||||||
|
|
||||||
const app = createApp(index)
|
|
||||||
|
|
||||||
app.use(
|
|
||||||
createVuetify({
|
|
||||||
directives,
|
|
||||||
components: {
|
|
||||||
...components,
|
|
||||||
...labsComponents,
|
|
||||||
},
|
|
||||||
blueprint: md3,
|
|
||||||
theme: {
|
|
||||||
defaultTheme: "original",
|
|
||||||
themes: {
|
|
||||||
original: {
|
|
||||||
colors: {
|
|
||||||
primary: "#4a5099",
|
|
||||||
secondary: "#2196f3",
|
|
||||||
accent: "#009688",
|
|
||||||
error: "#f44336",
|
|
||||||
warning: "#ff9800",
|
|
||||||
info: "#03a9f4",
|
|
||||||
success: "#4caf50",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
}),
|
|
||||||
)
|
|
||||||
|
|
||||||
app.use(createPinia())
|
|
||||||
app.use(router)
|
|
||||||
|
|
||||||
app.mount("#app")
|
|
@ -1,116 +0,0 @@
|
|||||||
import { createRouter, createWebHistory } from "vue-router"
|
|
||||||
import { useUserinfo } from "@/stores/userinfo"
|
|
||||||
import UserCenterLayout from "@/layouts/user-center.vue"
|
|
||||||
import AdministratorLayout from "@/layouts/administrator.vue"
|
|
||||||
|
|
||||||
const router = createRouter({
|
|
||||||
history: createWebHistory(import.meta.env.BASE_URL),
|
|
||||||
routes: [
|
|
||||||
{
|
|
||||||
path: "/",
|
|
||||||
redirect: { name: "dashboard" },
|
|
||||||
meta: { public: true },
|
|
||||||
},
|
|
||||||
{
|
|
||||||
path: "/users",
|
|
||||||
component: UserCenterLayout,
|
|
||||||
children: [
|
|
||||||
{
|
|
||||||
path: "/me",
|
|
||||||
name: "dashboard",
|
|
||||||
component: () => import("@/views/dashboard.vue"),
|
|
||||||
meta: { title: "Your account" },
|
|
||||||
},
|
|
||||||
{
|
|
||||||
path: "/me/personalize",
|
|
||||||
name: "personalize",
|
|
||||||
component: () => import("@/views/personalize.vue"),
|
|
||||||
meta: { title: "Your personality" },
|
|
||||||
},
|
|
||||||
{
|
|
||||||
path: "/me/security",
|
|
||||||
name: "security",
|
|
||||||
component: () => import("@/views/security.vue"),
|
|
||||||
meta: { title: "Your security" },
|
|
||||||
},
|
|
||||||
],
|
|
||||||
},
|
|
||||||
{
|
|
||||||
path: "/",
|
|
||||||
children: [
|
|
||||||
{
|
|
||||||
path: "/sign-in",
|
|
||||||
alias: ["/mfa"],
|
|
||||||
name: "auth.sign-in",
|
|
||||||
component: () => import("@/views/auth/sign-in.vue"),
|
|
||||||
meta: { public: true, title: "Sign in" },
|
|
||||||
},
|
|
||||||
{
|
|
||||||
path: "/sign-up",
|
|
||||||
name: "auth.sign-up",
|
|
||||||
component: () => import("@/views/auth/sign-up.vue"),
|
|
||||||
meta: { public: true, title: "Sign up" },
|
|
||||||
},
|
|
||||||
{
|
|
||||||
path: "/authorize",
|
|
||||||
name: "oauth.authorize",
|
|
||||||
component: () => import("@/views/auth/authorize.vue"),
|
|
||||||
},
|
|
||||||
],
|
|
||||||
},
|
|
||||||
{
|
|
||||||
path: "/flow",
|
|
||||||
children: [
|
|
||||||
{
|
|
||||||
path: "confirm",
|
|
||||||
name: "callback.confirm",
|
|
||||||
component: () => import("@/views/flow/confirm.vue"),
|
|
||||||
meta: { public: true, title: "Confirm registration" },
|
|
||||||
},
|
|
||||||
{
|
|
||||||
path: "password-reset",
|
|
||||||
name: "callback.password-reset",
|
|
||||||
component: () => import("@/views/flow/password-reset.vue"),
|
|
||||||
meta: { public: true, title: "Reset password" },
|
|
||||||
},
|
|
||||||
],
|
|
||||||
},
|
|
||||||
{
|
|
||||||
path: "/admin",
|
|
||||||
component: AdministratorLayout,
|
|
||||||
children: [
|
|
||||||
{
|
|
||||||
path: "",
|
|
||||||
name: "admin.dashboard",
|
|
||||||
component: () => import("@/views/admin/dashboard.vue"),
|
|
||||||
},
|
|
||||||
{
|
|
||||||
path: "users",
|
|
||||||
name: "admin.users",
|
|
||||||
component: () => import("@/views/admin/users.vue"),
|
|
||||||
},
|
|
||||||
]
|
|
||||||
}
|
|
||||||
],
|
|
||||||
})
|
|
||||||
|
|
||||||
router.beforeEach(async (to, from, next) => {
|
|
||||||
const id = useUserinfo()
|
|
||||||
if (!id.isReady) {
|
|
||||||
await id.readProfiles()
|
|
||||||
}
|
|
||||||
|
|
||||||
if (to.meta.title) {
|
|
||||||
document.title = `Solarpass | ${to.meta.title}`
|
|
||||||
} else {
|
|
||||||
document.title = "Solarpass"
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!to.meta.public && !id.userinfo.isLoggedIn) {
|
|
||||||
next({ name: "auth.sign-in", query: { redirect_uri: to.fullPath } })
|
|
||||||
} else {
|
|
||||||
next()
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
export default router
|
|
@ -1,3 +0,0 @@
|
|||||||
export async function request(input: string, init?: RequestInit) {
|
|
||||||
return await fetch(input, init)
|
|
||||||
}
|
|
@ -1,67 +0,0 @@
|
|||||||
import { defineStore } from "pinia"
|
|
||||||
import { ref } from "vue"
|
|
||||||
import { checkLoggedIn, getAtk } from "@/stores/userinfo"
|
|
||||||
import { request } from "@/scripts/request"
|
|
||||||
|
|
||||||
export const useNotifications = defineStore("notifications", () => {
|
|
||||||
let socket: WebSocket
|
|
||||||
|
|
||||||
const loading = ref(false)
|
|
||||||
|
|
||||||
const notifications = ref<any[]>([])
|
|
||||||
const total = ref(0)
|
|
||||||
|
|
||||||
async function list() {
|
|
||||||
loading.value = true
|
|
||||||
const res = await request(
|
|
||||||
"/api/notifications?" +
|
|
||||||
new URLSearchParams({
|
|
||||||
take: (25).toString(),
|
|
||||||
offset: (0).toString(),
|
|
||||||
}),
|
|
||||||
{
|
|
||||||
headers: { Authorization: `Bearer ${getAtk()}` },
|
|
||||||
},
|
|
||||||
)
|
|
||||||
if (res.status === 200) {
|
|
||||||
const data = await res.json()
|
|
||||||
notifications.value = data["data"]
|
|
||||||
total.value = data["count"]
|
|
||||||
}
|
|
||||||
loading.value = false
|
|
||||||
}
|
|
||||||
|
|
||||||
function remove(idx: number) {
|
|
||||||
notifications.value.splice(idx, 1)
|
|
||||||
total.value--
|
|
||||||
}
|
|
||||||
|
|
||||||
async function connect() {
|
|
||||||
if (!(checkLoggedIn())) return
|
|
||||||
|
|
||||||
const protocol = location.protocol.replace("http", "ws")
|
|
||||||
const uri = `${protocol}//${window.location.host}/api/ws`
|
|
||||||
|
|
||||||
socket = new WebSocket(uri + `?tk=${getAtk() as string}`)
|
|
||||||
|
|
||||||
socket.addEventListener("open", (event) => {
|
|
||||||
console.log("[NOTIFICATIONS] The listen websocket has been established... ", event.type)
|
|
||||||
})
|
|
||||||
socket.addEventListener("close", (event) => {
|
|
||||||
console.warn("[NOTIFICATIONS] The listen websocket is disconnected... ", event.reason, event.code)
|
|
||||||
})
|
|
||||||
socket.addEventListener("message", (event) => {
|
|
||||||
const data = JSON.parse(event.data)
|
|
||||||
if (data["w"] == "notifications.new") {
|
|
||||||
notifications.value.push(data["p"])
|
|
||||||
total.value++
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
function disconnect() {
|
|
||||||
socket.close()
|
|
||||||
}
|
|
||||||
|
|
||||||
return { loading, notifications, total, list, remove, connect, disconnect }
|
|
||||||
})
|
|
@ -1,54 +0,0 @@
|
|||||||
import Cookie from "universal-cookie"
|
|
||||||
import { defineStore } from "pinia"
|
|
||||||
import { ref } from "vue"
|
|
||||||
import { request } from "@/scripts/request"
|
|
||||||
|
|
||||||
export interface Userinfo {
|
|
||||||
isLoggedIn: boolean
|
|
||||||
displayName: string
|
|
||||||
data: any
|
|
||||||
}
|
|
||||||
|
|
||||||
export const defaultUserinfo: Userinfo = {
|
|
||||||
isLoggedIn: false,
|
|
||||||
displayName: "Citizen",
|
|
||||||
data: null,
|
|
||||||
}
|
|
||||||
|
|
||||||
export function getAtk(): string {
|
|
||||||
return new Cookie().get("__hydrogen_atk")
|
|
||||||
}
|
|
||||||
|
|
||||||
export function checkLoggedIn(): boolean {
|
|
||||||
return new Cookie().get("__hydrogen_rtk")
|
|
||||||
}
|
|
||||||
|
|
||||||
export const useUserinfo = defineStore("userinfo", () => {
|
|
||||||
const userinfo = ref(defaultUserinfo)
|
|
||||||
const isReady = ref(false)
|
|
||||||
|
|
||||||
async function readProfiles() {
|
|
||||||
if (!checkLoggedIn()) {
|
|
||||||
isReady.value = true
|
|
||||||
}
|
|
||||||
|
|
||||||
const res = await request("/api/users/me", {
|
|
||||||
headers: { Authorization: `Bearer ${getAtk()}` },
|
|
||||||
})
|
|
||||||
|
|
||||||
if (res.status !== 200) {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
const data = await res.json()
|
|
||||||
|
|
||||||
isReady.value = true
|
|
||||||
userinfo.value = {
|
|
||||||
isLoggedIn: true,
|
|
||||||
displayName: data["nick"],
|
|
||||||
data: data,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return { userinfo, isReady, readProfiles }
|
|
||||||
})
|
|
@ -1,51 +0,0 @@
|
|||||||
<template>
|
|
||||||
<div class="w-full h-full flex justify-center items-center">
|
|
||||||
<v-empty-state
|
|
||||||
headline="Administration"
|
|
||||||
icon="mdi-cog"
|
|
||||||
title="What would you like to do today?"
|
|
||||||
>
|
|
||||||
<v-container>
|
|
||||||
<v-row>
|
|
||||||
<v-col cols="12" md="6">
|
|
||||||
<v-card
|
|
||||||
:to="{ name: 'admin.users' }"
|
|
||||||
prepend-icon="mdi-account-group"
|
|
||||||
text="Manage to help users do something they can't"
|
|
||||||
title="Users"
|
|
||||||
></v-card>
|
|
||||||
</v-col>
|
|
||||||
|
|
||||||
<v-col cols="12" md="6">
|
|
||||||
<v-card
|
|
||||||
disabled
|
|
||||||
prepend-icon="mdi-comment-quote"
|
|
||||||
text="Manage the content on the platform"
|
|
||||||
title="Posts & Articles"
|
|
||||||
></v-card>
|
|
||||||
</v-col>
|
|
||||||
|
|
||||||
<v-col cols="12" md="6">
|
|
||||||
<v-card
|
|
||||||
disabled
|
|
||||||
prepend-icon="mdi-file-cabinet"
|
|
||||||
text="Manage attachments on the platform"
|
|
||||||
title="Attachments"
|
|
||||||
></v-card>
|
|
||||||
</v-col>
|
|
||||||
|
|
||||||
<v-col cols="12" md="6">
|
|
||||||
<v-card
|
|
||||||
disabled
|
|
||||||
prepend-icon="mdi-ticket"
|
|
||||||
text="Solve the tickets issued by users"
|
|
||||||
title="Tickets"
|
|
||||||
></v-card>
|
|
||||||
</v-col>
|
|
||||||
</v-row>
|
|
||||||
</v-container>
|
|
||||||
</v-empty-state>
|
|
||||||
</div>
|
|
||||||
</template>
|
|
||||||
<script setup lang="ts">
|
|
||||||
</script>
|
|
@ -1,142 +0,0 @@
|
|||||||
<template>
|
|
||||||
<div>
|
|
||||||
<v-data-table-server
|
|
||||||
fixed-header
|
|
||||||
class="h-full"
|
|
||||||
density="compact"
|
|
||||||
:headers="dataDefinitions.users"
|
|
||||||
:items="users"
|
|
||||||
:items-length="pagination.total"
|
|
||||||
:loading="reverting"
|
|
||||||
v-model:items-per-page="pagination.pageSize"
|
|
||||||
@update:options="readUsers"
|
|
||||||
item-value="id"
|
|
||||||
>
|
|
||||||
<template v-slot:top>
|
|
||||||
<v-toolbar color="secondary">
|
|
||||||
<div class="max-md:px-5 md:px-12 flex flex-grow-1 items-center">
|
|
||||||
<v-btn class="me-2" icon="mdi-account-group" density="compact" :to="{ name: 'admin.dashboard' }" exact />
|
|
||||||
<h3 class="ml-2 text-lg font-500">Users</h3>
|
|
||||||
</div>
|
|
||||||
</v-toolbar>
|
|
||||||
</template>
|
|
||||||
|
|
||||||
<template v-slot:item="{ item }: { item: any }">
|
|
||||||
<tr>
|
|
||||||
<td>{{ item.id }}</td>
|
|
||||||
<td>{{ item.name }}</td>
|
|
||||||
<td>{{ item.nick }}</td>
|
|
||||||
<td>{{ new Date(item.created_at).toLocaleString() }}</td>
|
|
||||||
<td>
|
|
||||||
<v-tooltip text="Details">
|
|
||||||
<template #activator="{ props }">
|
|
||||||
<v-btn
|
|
||||||
v-bind="props"
|
|
||||||
variant="text"
|
|
||||||
size="x-small"
|
|
||||||
color="info"
|
|
||||||
icon="mdi-dots-horizontal"
|
|
||||||
@click="viewingUser = item"
|
|
||||||
/>
|
|
||||||
</template>
|
|
||||||
</v-tooltip>
|
|
||||||
<v-tooltip text="Assign Permissions">
|
|
||||||
<template #activator="{ props }">
|
|
||||||
<v-btn
|
|
||||||
v-bind="props"
|
|
||||||
variant="text"
|
|
||||||
size="x-small"
|
|
||||||
color="teal"
|
|
||||||
icon="mdi-code-block-braces"
|
|
||||||
@click="assigningPermUser = item"
|
|
||||||
/>
|
|
||||||
</template>
|
|
||||||
</v-tooltip>
|
|
||||||
<v-tooltip text="View Auth Factors">
|
|
||||||
<template #activator="{ props }">
|
|
||||||
<v-btn
|
|
||||||
v-bind="props"
|
|
||||||
variant="text"
|
|
||||||
size="x-small"
|
|
||||||
color="warning"
|
|
||||||
icon="mdi-lock"
|
|
||||||
@click="viewingFactorUser = item"
|
|
||||||
/>
|
|
||||||
</template>
|
|
||||||
</v-tooltip>
|
|
||||||
</td>
|
|
||||||
</tr>
|
|
||||||
</template>
|
|
||||||
</v-data-table-server>
|
|
||||||
|
|
||||||
<user-detail-panel :data="viewingUser" @close="viewingUser = null" />
|
|
||||||
<user-assign-perms-panel :data="assigningPermUser" @close="assigningPermUser = null"
|
|
||||||
@success="readUsers(pagination)"
|
|
||||||
@error="(val: string) => error = val" />
|
|
||||||
<user-factor-panel :data="viewingFactorUser" @close="viewingFactorUser = null"
|
|
||||||
@error="(val: string) => error = val" />
|
|
||||||
|
|
||||||
<v-snackbar :timeout="3000" :model-value="error != null" @update:model-value="() => error = null">
|
|
||||||
{{ error }}
|
|
||||||
</v-snackbar>
|
|
||||||
</div>
|
|
||||||
</template>
|
|
||||||
|
|
||||||
<script setup lang="ts">
|
|
||||||
import { onMounted, reactive, ref } from "vue"
|
|
||||||
import { request } from "@/scripts/request"
|
|
||||||
import { getAtk } from "@/stores/userinfo"
|
|
||||||
import UserDetailPanel from "@/components/admin/UserDetailPanel.vue"
|
|
||||||
import UserAssignPermsPanel from "@/components/admin/UserAssignPermsPanel.vue"
|
|
||||||
import UserFactorPanel from "@/components/admin/UserFactorPanel.vue"
|
|
||||||
|
|
||||||
const error = ref<string | null>(null)
|
|
||||||
|
|
||||||
const users = ref<any[]>([])
|
|
||||||
|
|
||||||
const viewingUser = ref<any>(null)
|
|
||||||
const viewingFactorUser = ref<any>(null)
|
|
||||||
const assigningPermUser = ref<any>(null)
|
|
||||||
|
|
||||||
const dataDefinitions: { [id: string]: any[] } = {
|
|
||||||
users: [
|
|
||||||
{ align: "start", key: "id", title: "ID" },
|
|
||||||
{ align: "start", key: "name", title: "Name" },
|
|
||||||
{ align: "start", key: "nick", title: "Nick" },
|
|
||||||
{ align: "start", key: "created_at", title: "Created At" },
|
|
||||||
{ align: "start", key: "actions", title: "Actions", sortable: false },
|
|
||||||
],
|
|
||||||
}
|
|
||||||
|
|
||||||
const reverting = ref(true)
|
|
||||||
const pagination = reactive({
|
|
||||||
page: 1, pageSize: 5, total: 0,
|
|
||||||
})
|
|
||||||
|
|
||||||
async function readUsers({ page, itemsPerPage }: { page?: number; itemsPerPage?: number }) {
|
|
||||||
if (itemsPerPage) pagination.pageSize = itemsPerPage
|
|
||||||
if (page) pagination.page = page
|
|
||||||
|
|
||||||
reverting.value = true
|
|
||||||
const res = await request(
|
|
||||||
"/api/admin/users?" +
|
|
||||||
new URLSearchParams({
|
|
||||||
take: pagination.pageSize.toString(),
|
|
||||||
offset: ((pagination.page - 1) * pagination.pageSize).toString(),
|
|
||||||
}),
|
|
||||||
{
|
|
||||||
headers: { Authorization: `Bearer ${getAtk()}` },
|
|
||||||
},
|
|
||||||
)
|
|
||||||
if (res.status !== 200) {
|
|
||||||
error.value = await res.text()
|
|
||||||
} else {
|
|
||||||
const data = await res.json()
|
|
||||||
users.value = data["data"]
|
|
||||||
pagination.total = data["count"]
|
|
||||||
}
|
|
||||||
reverting.value = false
|
|
||||||
}
|
|
||||||
|
|
||||||
onMounted(() => readUsers({}))
|
|
||||||
</script>
|
|
@ -1,189 +0,0 @@
|
|||||||
<template>
|
|
||||||
<v-container class="h-screen flex flex-col gap-3 items-center justify-center">
|
|
||||||
<v-card class="w-full max-w-[720px] overflow-auto" :loading="loading">
|
|
||||||
<v-card-text class="card-grid pa-9">
|
|
||||||
<div>
|
|
||||||
<v-avatar color="accent" icon="mdi-connection" size="large" class="card-rounded mb-2" />
|
|
||||||
<h1 class="text-2xl">Connect to third-party</h1>
|
|
||||||
<p>One Solarpass, entire internet.</p>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<v-window :touch="false" :model-value="panel" class="pa-2 mx-[-0.5rem]">
|
|
||||||
<v-window-item value="confirm">
|
|
||||||
<div class="flex flex-col gap-2">
|
|
||||||
<v-expand-transition>
|
|
||||||
<v-alert v-show="error" variant="tonal" type="error" class="text-xs mb-3">
|
|
||||||
<p>Something went wrong... {{ error }}</p>
|
|
||||||
<br />
|
|
||||||
|
|
||||||
<p class="font-bold">
|
|
||||||
It's usually not our fault. Try bringing this link to give feedback to the developer of the app you
|
|
||||||
came from.
|
|
||||||
</p>
|
|
||||||
</v-alert>
|
|
||||||
</v-expand-transition>
|
|
||||||
|
|
||||||
<div v-if="!error">
|
|
||||||
<h1 class="font-bold text-xl">{{ metadata?.name ?? "Loading" }}</h1>
|
|
||||||
<p>{{ metadata?.description ?? "Hold on a second please!" }}</p>
|
|
||||||
|
|
||||||
<div class="mt-3">
|
|
||||||
<p class="opacity-80 text-xs">Permissions they requested</p>
|
|
||||||
<v-card variant="tonal" class="mt-1 mx-[-4px]">
|
|
||||||
<v-list density="compact">
|
|
||||||
<v-list-item v-for="(claim, key) in requestedClaims" :key="key" lines="two">
|
|
||||||
<template #title>
|
|
||||||
<span class="capitalize">{{ getClaimDescription(claim)?.name }}</span>
|
|
||||||
</template>
|
|
||||||
<template #subtitle>
|
|
||||||
<span>{{ getClaimDescription(claim)?.description }}</span>
|
|
||||||
</template>
|
|
||||||
<template #prepend>
|
|
||||||
<v-icon :icon="getClaimDescription(claim)?.icon" size="x-large" />
|
|
||||||
</template>
|
|
||||||
</v-list-item>
|
|
||||||
</v-list>
|
|
||||||
</v-card>
|
|
||||||
|
|
||||||
<div class="mt-5 flex justify-between">
|
|
||||||
<v-btn prepend-icon="mdi-close" variant="text" color="error" :disabled="loading" @click="decline">
|
|
||||||
Decline
|
|
||||||
</v-btn>
|
|
||||||
<v-btn append-icon="mdi-check" variant="tonal" color="success" :disabled="loading" @click="approve">
|
|
||||||
Approve
|
|
||||||
</v-btn>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="mt-5 text-xs text-center opacity-75">
|
|
||||||
<p>After approve their request, you will be redirect to</p>
|
|
||||||
<p class="text-mono">{{ route.query["redirect_uri"] }}</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</v-window-item>
|
|
||||||
|
|
||||||
<v-window-item value="callback">
|
|
||||||
<div>
|
|
||||||
<v-icon icon="mdi-fire" size="32" color="grey-darken-3" class="mb-3" />
|
|
||||||
|
|
||||||
<h1 class="font-bold text-xl">Authoirzed</h1>
|
|
||||||
<p>You're done! We sucessfully established connection between you and {{ metadata?.name }}.</p>
|
|
||||||
|
|
||||||
<p class="mt-3">Now you can continue your their app, we will redirect you soon.</p>
|
|
||||||
|
|
||||||
<p class="mt-3">Teleporting you to...</p>
|
|
||||||
<p class="text-xs text-mono">{{ route.query["redirect_uri"] }}</p>
|
|
||||||
</div>
|
|
||||||
</v-window-item>
|
|
||||||
</v-window>
|
|
||||||
</v-card-text>
|
|
||||||
</v-card>
|
|
||||||
|
|
||||||
<copyright />
|
|
||||||
</v-container>
|
|
||||||
</template>
|
|
||||||
|
|
||||||
<script setup lang="ts">
|
|
||||||
import { computed, ref } from "vue"
|
|
||||||
import { useRoute } from "vue-router"
|
|
||||||
import { request } from "@/scripts/request"
|
|
||||||
import { getAtk } from "@/stores/userinfo"
|
|
||||||
import { claims, type ClaimType } from "@/views/auth/claims"
|
|
||||||
import Copyright from "@/components/Copyright.vue"
|
|
||||||
|
|
||||||
const route = useRoute()
|
|
||||||
|
|
||||||
const error = ref<string | null>(null)
|
|
||||||
|
|
||||||
const loading = ref(false)
|
|
||||||
|
|
||||||
const metadata = ref<any>(null)
|
|
||||||
const requestedClaims = computed(() => {
|
|
||||||
const scope: string = (route.query["scope"] as string) ?? ""
|
|
||||||
return scope.split(" ")
|
|
||||||
})
|
|
||||||
|
|
||||||
const panel = ref("confirm")
|
|
||||||
|
|
||||||
async function tryAuthorize() {
|
|
||||||
const res = await request("/api/auth/o/authorize" + window.location.search, {
|
|
||||||
headers: { Authorization: `Bearer ${getAtk()}` },
|
|
||||||
})
|
|
||||||
|
|
||||||
if (res.status !== 200) {
|
|
||||||
error.value = await res.text()
|
|
||||||
} else {
|
|
||||||
const data = await res.json()
|
|
||||||
|
|
||||||
if (data["ticket"]) {
|
|
||||||
panel.value = "callback"
|
|
||||||
callback(data["ticket"])
|
|
||||||
} else {
|
|
||||||
document.title = `Solarpass | Connect to ${data["client"]?.name}`
|
|
||||||
metadata.value = data["client"]
|
|
||||||
loading.value = false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
tryAuthorize()
|
|
||||||
|
|
||||||
function decline() {
|
|
||||||
if (window.history.length > 0) {
|
|
||||||
window.history.back()
|
|
||||||
} else {
|
|
||||||
window.close()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async function approve() {
|
|
||||||
loading.value = true
|
|
||||||
const res = await request("/api/auth/o/authorize" + window.location.search, {
|
|
||||||
method: "POST",
|
|
||||||
headers: { Authorization: `Bearer ${getAtk()}` },
|
|
||||||
})
|
|
||||||
|
|
||||||
if (res.status !== 200) {
|
|
||||||
error.value = await res.text()
|
|
||||||
loading.value = false
|
|
||||||
} else {
|
|
||||||
const data = await res.json()
|
|
||||||
panel.value = "callback"
|
|
||||||
setTimeout(() => callback(data["ticket"]), 1850)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function callback(ticket: any) {
|
|
||||||
const url = `${route.query["redirect_uri"]}?code=${ticket["grant_token"]}&state=${route.query["state"]}`
|
|
||||||
window.open(url, "_self")
|
|
||||||
}
|
|
||||||
|
|
||||||
function getClaimDescription(key: string): ClaimType {
|
|
||||||
return Object.prototype.hasOwnProperty.call(claims, key)
|
|
||||||
? claims[key]
|
|
||||||
: {
|
|
||||||
icon: "mdi-asterisk",
|
|
||||||
name: key,
|
|
||||||
description: "Unknown claim...",
|
|
||||||
}
|
|
||||||
}
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<style scoped>
|
|
||||||
.card-grid {
|
|
||||||
display: grid;
|
|
||||||
grid-template-columns: 1fr 1fr;
|
|
||||||
gap: 1rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
@media (max-width: 768px) {
|
|
||||||
.card-grid {
|
|
||||||
grid-template-columns: 1fr;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
.card-rounded {
|
|
||||||
border-radius: 8px;
|
|
||||||
}
|
|
||||||
</style>
|
|
@ -1,13 +0,0 @@
|
|||||||
export interface ClaimType {
|
|
||||||
icon: string
|
|
||||||
name: string
|
|
||||||
description: string
|
|
||||||
}
|
|
||||||
|
|
||||||
export const claims: { [id: string]: ClaimType } = {
|
|
||||||
openid: {
|
|
||||||
icon: "mdi-identifier",
|
|
||||||
name: "Open Identity",
|
|
||||||
description: "Allow them to read your personal information.",
|
|
||||||
},
|
|
||||||
}
|
|
@ -1,85 +0,0 @@
|
|||||||
<template>
|
|
||||||
<v-container class="h-screen flex flex-col gap-3 items-center justify-center">
|
|
||||||
<callback-notify />
|
|
||||||
|
|
||||||
<v-card class="w-full max-w-[720px] overflow-auto" :loading="loading">
|
|
||||||
<v-card-text class="card-grid pa-9">
|
|
||||||
<div>
|
|
||||||
<v-avatar color="accent" icon="mdi-login-variant" size="large" class="card-rounded mb-2" />
|
|
||||||
<h1 class="text-2xl">Sign in</h1>
|
|
||||||
<p v-if="ticket">We need to verify that the person trying to access your account is you.</p>
|
|
||||||
<p v-else>Sign in via your Solar ID to access the entire Solar Network.</p>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<v-window :touch="false" :model-value="panel" class="pa-2 mx-[-0.5rem]">
|
|
||||||
<v-window-item v-for="(k, idx) in Object.keys(panels)" :key="idx" :value="k">
|
|
||||||
<component :is="panels[k]" @swap="(val: string) => (panel = val)" v-model:loading="loading"
|
|
||||||
v-model:currentFactor="currentFactor" v-model:ticket="ticket" />
|
|
||||||
</v-window-item>
|
|
||||||
</v-window>
|
|
||||||
</v-card-text>
|
|
||||||
</v-card>
|
|
||||||
|
|
||||||
<copyright />
|
|
||||||
</v-container>
|
|
||||||
</template>
|
|
||||||
|
|
||||||
<script setup lang="ts">
|
|
||||||
import { type Component, onMounted, ref } from "vue"
|
|
||||||
import { useRoute } from "vue-router"
|
|
||||||
import Copyright from "@/components/Copyright.vue"
|
|
||||||
import CallbackNotify from "@/components/auth/CallbackNotify.vue"
|
|
||||||
import FactorPicker from "@/components/auth/FactorPicker.vue"
|
|
||||||
import FactorApplicator from "@/components/auth/FactorApplicator.vue"
|
|
||||||
import AccountAuthenticate from "@/components/auth/Authenticate.vue"
|
|
||||||
import AuthenticateCompleted from "@/components/auth/AuthenticateCompleted.vue"
|
|
||||||
|
|
||||||
const route = useRoute()
|
|
||||||
|
|
||||||
const loading = ref(false)
|
|
||||||
|
|
||||||
const currentFactor = ref<any>(null)
|
|
||||||
const ticket = ref<any>(null)
|
|
||||||
|
|
||||||
async function pickUpTicket() {
|
|
||||||
if (route.query["ticketId"]) {
|
|
||||||
loading.value = true
|
|
||||||
const res = await fetch(`/api/auth/tickets/${route.query["ticketId"]}`)
|
|
||||||
if (res.status == 200) {
|
|
||||||
ticket.value = await res.json()
|
|
||||||
if (ticket.value["available_at"] != null) panel.value = "completed"
|
|
||||||
else panel.value = "mfa"
|
|
||||||
}
|
|
||||||
loading.value = false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
onMounted(() => pickUpTicket())
|
|
||||||
|
|
||||||
const panel = ref("authenticate")
|
|
||||||
|
|
||||||
const panels: { [id: string]: Component } = {
|
|
||||||
authenticate: AccountAuthenticate,
|
|
||||||
mfa: FactorPicker,
|
|
||||||
applicator: FactorApplicator,
|
|
||||||
completed: AuthenticateCompleted,
|
|
||||||
}
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<style scoped>
|
|
||||||
.card-grid {
|
|
||||||
display: grid;
|
|
||||||
grid-template-columns: 1fr 1fr;
|
|
||||||
gap: 1rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
@media (max-width: 768px) {
|
|
||||||
.card-grid {
|
|
||||||
grid-template-columns: 1fr;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
.card-rounded {
|
|
||||||
border-radius: 8px;
|
|
||||||
}
|
|
||||||
</style>
|
|
@ -1,162 +0,0 @@
|
|||||||
<template>
|
|
||||||
<v-container class="h-screen flex flex-col gap-3 items-center justify-center">
|
|
||||||
<callback-notify />
|
|
||||||
|
|
||||||
<v-card class="w-full max-w-[720px] overflow-auto" :loading="loading">
|
|
||||||
<v-card-text class="card-grid pa-9">
|
|
||||||
<div>
|
|
||||||
<v-avatar color="accent" icon="mdi-login-variant" size="large" class="card-rounded mb-2" />
|
|
||||||
<h1 class="text-2xl">Create an account</h1>
|
|
||||||
<p>Create an account on Solar Network. Then enjoy all our services.</p>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="flex items-center">
|
|
||||||
<v-form class="flex-grow-1" @submit.prevent="submit">
|
|
||||||
<v-row dense class="mb-3">
|
|
||||||
<v-col :cols="6">
|
|
||||||
<v-text-field
|
|
||||||
hide-details
|
|
||||||
label="Name"
|
|
||||||
autocomplete="username"
|
|
||||||
variant="solo"
|
|
||||||
density="comfortable"
|
|
||||||
v-model="data.name"
|
|
||||||
/>
|
|
||||||
</v-col>
|
|
||||||
<v-col :cols="6">
|
|
||||||
<v-text-field
|
|
||||||
hide-details
|
|
||||||
label="Nick"
|
|
||||||
autocomplete="nickname"
|
|
||||||
variant="solo"
|
|
||||||
density="comfortable"
|
|
||||||
v-model="data.nick"
|
|
||||||
/>
|
|
||||||
</v-col>
|
|
||||||
<v-col :cols="12">
|
|
||||||
<v-text-field
|
|
||||||
hide-details
|
|
||||||
label="Email Address"
|
|
||||||
type="email"
|
|
||||||
variant="solo"
|
|
||||||
density="comfortable"
|
|
||||||
v-model="data.email"
|
|
||||||
/>
|
|
||||||
</v-col>
|
|
||||||
<v-col :cols="12">
|
|
||||||
<v-text-field
|
|
||||||
hide-details
|
|
||||||
label="Password"
|
|
||||||
type="password"
|
|
||||||
autocomplete="new-password"
|
|
||||||
variant="solo"
|
|
||||||
density="comfortable"
|
|
||||||
v-model="data.password"
|
|
||||||
/>
|
|
||||||
</v-col>
|
|
||||||
</v-row>
|
|
||||||
|
|
||||||
<v-expand-transition>
|
|
||||||
<v-alert v-show="error" variant="tonal" type="error" class="text-xs mb-3">
|
|
||||||
Something went wrong... {{ error }}
|
|
||||||
</v-alert>
|
|
||||||
</v-expand-transition>
|
|
||||||
|
|
||||||
<div class="flex justify-between">
|
|
||||||
<v-btn type="button" variant="plain" color="grey-darken-3" :to="{ name: 'auth.sign-in' }">
|
|
||||||
Sign in
|
|
||||||
</v-btn>
|
|
||||||
|
|
||||||
<v-btn type="submit" variant="text" color="primary" append-icon="mdi-arrow-right" :disabled="loading">
|
|
||||||
Next
|
|
||||||
</v-btn>
|
|
||||||
</div>
|
|
||||||
</v-form>
|
|
||||||
</div>
|
|
||||||
</v-card-text>
|
|
||||||
</v-card>
|
|
||||||
|
|
||||||
<v-dialog v-model="done" class="max-w-[560px]">
|
|
||||||
<v-card title="Congratulations">
|
|
||||||
<template #text>
|
|
||||||
You successfully created an account on Solar Network. Now sign in to your account and start exploring!
|
|
||||||
</template>
|
|
||||||
<template #actions>
|
|
||||||
<div class="flex flex-grow-1 justify-end">
|
|
||||||
<v-btn @click="callback">Let's go</v-btn>
|
|
||||||
</div>
|
|
||||||
</template>
|
|
||||||
</v-card>
|
|
||||||
</v-dialog>
|
|
||||||
|
|
||||||
<copyright />
|
|
||||||
</v-container>
|
|
||||||
</template>
|
|
||||||
|
|
||||||
<script setup lang="ts">
|
|
||||||
import { ref } from "vue"
|
|
||||||
import { request } from "@/scripts/request"
|
|
||||||
import { useRoute, useRouter } from "vue-router"
|
|
||||||
import Copyright from "@/components/Copyright.vue"
|
|
||||||
import CallbackNotify from "@/components/auth/CallbackNotify.vue"
|
|
||||||
|
|
||||||
const error = ref<string | null>(null)
|
|
||||||
|
|
||||||
const route = useRoute()
|
|
||||||
const router = useRouter()
|
|
||||||
|
|
||||||
const done = ref(false)
|
|
||||||
const loading = ref(false)
|
|
||||||
|
|
||||||
const data = ref({
|
|
||||||
name: "",
|
|
||||||
nick: "",
|
|
||||||
email: "",
|
|
||||||
password: "",
|
|
||||||
})
|
|
||||||
|
|
||||||
async function submit() {
|
|
||||||
const payload = data.value
|
|
||||||
if (!payload.name || !payload.nick || !payload.email || !payload.password) return
|
|
||||||
|
|
||||||
loading.value = true
|
|
||||||
const res = await request("/api/users", {
|
|
||||||
method: "POST",
|
|
||||||
headers: { "Content-Type": "application/json" },
|
|
||||||
body: JSON.stringify(payload),
|
|
||||||
})
|
|
||||||
if (res.status !== 200) {
|
|
||||||
error.value = await res.text()
|
|
||||||
} else {
|
|
||||||
done.value = true
|
|
||||||
error.value = null
|
|
||||||
}
|
|
||||||
loading.value = false
|
|
||||||
}
|
|
||||||
|
|
||||||
function callback() {
|
|
||||||
if (route.params["closable"]) {
|
|
||||||
window.close()
|
|
||||||
} else {
|
|
||||||
router.push({ name: "auth.sign-in" })
|
|
||||||
}
|
|
||||||
}
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<style scoped>
|
|
||||||
.card-grid {
|
|
||||||
display: grid;
|
|
||||||
grid-template-columns: 1fr 1fr;
|
|
||||||
gap: 1rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
@media (max-width: 768px) {
|
|
||||||
.card-grid {
|
|
||||||
grid-template-columns: 1fr;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
.card-rounded {
|
|
||||||
border-radius: 8px;
|
|
||||||
}
|
|
||||||
</style>
|
|
@ -1,77 +0,0 @@
|
|||||||
<template>
|
|
||||||
<div>
|
|
||||||
<v-card>
|
|
||||||
<v-img cover class="bg-grey-lighten-2" :height="240" src="/api/users/me/banner" />
|
|
||||||
|
|
||||||
<v-card-text class="flex gap-3.5 px-5 pb-5">
|
|
||||||
<v-avatar
|
|
||||||
color="grey-lighten-2"
|
|
||||||
icon="mdi-account-circle"
|
|
||||||
class="rounded-card"
|
|
||||||
image="/api/users/me/avatar"
|
|
||||||
:size="54"
|
|
||||||
/>
|
|
||||||
<div>
|
|
||||||
<h1 class="text-2xl cursor-pointer" @click="show.realname = !show.realname">{{ displayName }}</h1>
|
|
||||||
<p v-html="description"></p>
|
|
||||||
|
|
||||||
<div class="mt-5">
|
|
||||||
<p class="opacity-80 desc-line">
|
|
||||||
<v-icon icon="mdi-calendar-blank" size="16" />
|
|
||||||
<span>Joined at {{ new Date(id.userinfo.data?.created_at)?.toLocaleString() }}</span>
|
|
||||||
</p>
|
|
||||||
<p class="opacity-80 desc-line">
|
|
||||||
<v-icon icon="mdi-cake-variant" size="16" />
|
|
||||||
<span>Birthday is {{ new Date(id.userinfo.data?.profile.birthday)?.toLocaleString() }}</span>
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</v-card-text>
|
|
||||||
</v-card>
|
|
||||||
</div>
|
|
||||||
</template>
|
|
||||||
|
|
||||||
<script setup lang="ts">
|
|
||||||
import { useUserinfo } from "@/stores/userinfo"
|
|
||||||
import { computed } from "vue"
|
|
||||||
import { reactive } from "vue"
|
|
||||||
import { parse } from "marked"
|
|
||||||
import dompurify from "dompurify"
|
|
||||||
|
|
||||||
const id = useUserinfo()
|
|
||||||
|
|
||||||
const displayName = computed(() => {
|
|
||||||
if (show.realname) {
|
|
||||||
return (
|
|
||||||
(id.userinfo.data?.profile?.first_name ?? "Unknown") + " " + (id.userinfo.data?.profile?.last_name ?? "Unknown")
|
|
||||||
)
|
|
||||||
} else {
|
|
||||||
return id.userinfo.displayName
|
|
||||||
}
|
|
||||||
})
|
|
||||||
const description = computed(() => {
|
|
||||||
if (id.userinfo.data?.description) {
|
|
||||||
return dompurify().sanitize(parse(id.userinfo.data?.description) as string)
|
|
||||||
} else {
|
|
||||||
return "No description yet."
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
const show = reactive({
|
|
||||||
realname: false,
|
|
||||||
})
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<style scoped>
|
|
||||||
.desc-line {
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
gap: 4px;
|
|
||||||
}
|
|
||||||
</style>
|
|
||||||
|
|
||||||
<style>
|
|
||||||
.rounded-card {
|
|
||||||
border-radius: 8px;
|
|
||||||
}
|
|
||||||
</style>
|
|
@ -1,104 +0,0 @@
|
|||||||
<template>
|
|
||||||
<v-container class="h-screen flex flex-col gap-3 items-center justify-center">
|
|
||||||
<v-card class="w-full max-w-[720px] overflow-auto" :loading="loading">
|
|
||||||
<v-card-text class="card-grid pa-9">
|
|
||||||
<div>
|
|
||||||
<v-avatar color="accent" icon="mdi-check-decagram" size="large" class="card-rounded mb-2" />
|
|
||||||
<h1 class="text-2xl">Confirm registration</h1>
|
|
||||||
<p>Confirm your account to unlock more abilities.</p>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<v-window :touch="false" :model-value="panel" class="pa-2 mx-[-0.5rem]">
|
|
||||||
<v-window-item value="confirm">
|
|
||||||
<div>
|
|
||||||
<v-expand-transition>
|
|
||||||
<v-alert v-show="error" variant="tonal" type="error" class="text-xs mb-3">
|
|
||||||
Something went wrong... {{ error }}
|
|
||||||
</v-alert>
|
|
||||||
</v-expand-transition>
|
|
||||||
|
|
||||||
<v-progress-circular v-if="!error" indeterminate size="32" color="grey-darken-3" class="mb-3" />
|
|
||||||
|
|
||||||
<h1 class="font-bold text-xl">Confirming</h1>
|
|
||||||
<p>We are confirming your account. Please stand by, this won't took a long time...</p>
|
|
||||||
</div>
|
|
||||||
</v-window-item>
|
|
||||||
<v-window-item value="callback">
|
|
||||||
<div>
|
|
||||||
<v-icon icon="mdi-fire" size="32" color="grey-darken-3" class="mb-3" />
|
|
||||||
|
|
||||||
<h1 class="font-bold text-xl">Confirmed</h1>
|
|
||||||
<p>You're done! We successfully confirmed your account.</p>
|
|
||||||
|
|
||||||
<p class="mt-3">Now you can continue to use Solarpass, we will redirect you to dashboard soon.</p>
|
|
||||||
</div>
|
|
||||||
</v-window-item>
|
|
||||||
</v-window>
|
|
||||||
</v-card-text>
|
|
||||||
</v-card>
|
|
||||||
|
|
||||||
<copyright />
|
|
||||||
</v-container>
|
|
||||||
</template>
|
|
||||||
|
|
||||||
<script setup lang="ts">
|
|
||||||
import { ref } from "vue"
|
|
||||||
import { useRoute, useRouter } from "vue-router"
|
|
||||||
import { request } from "@/scripts/request"
|
|
||||||
import { useUserinfo } from "@/stores/userinfo"
|
|
||||||
import Copyright from "@/components/Copyright.vue"
|
|
||||||
|
|
||||||
const route = useRoute()
|
|
||||||
const router = useRouter()
|
|
||||||
const { readProfiles } = useUserinfo()
|
|
||||||
|
|
||||||
const error = ref<string | null>(null)
|
|
||||||
|
|
||||||
const loading = ref(false)
|
|
||||||
|
|
||||||
const panel = ref("confirm")
|
|
||||||
|
|
||||||
async function confirm() {
|
|
||||||
if (!route.query["code"]) {
|
|
||||||
error.value = "code was not exists"
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
const res = await request("/api/users/me/confirm", {
|
|
||||||
method: "POST",
|
|
||||||
headers: { "Content-Type": "application/json" },
|
|
||||||
body: JSON.stringify({
|
|
||||||
code: route.query["code"],
|
|
||||||
}),
|
|
||||||
})
|
|
||||||
if (res.status !== 200) {
|
|
||||||
error.value = await res.text()
|
|
||||||
} else {
|
|
||||||
loading.value = true
|
|
||||||
panel.value = "callback"
|
|
||||||
await readProfiles()
|
|
||||||
await router.push({ name: "dashboard" })
|
|
||||||
}
|
|
||||||
loading.value = false
|
|
||||||
}
|
|
||||||
|
|
||||||
confirm()
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<style scoped>
|
|
||||||
.card-grid {
|
|
||||||
display: grid;
|
|
||||||
grid-template-columns: 1fr 1fr;
|
|
||||||
gap: 1rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
@media (max-width: 768px) {
|
|
||||||
.card-grid {
|
|
||||||
grid-template-columns: 1fr;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
.card-rounded {
|
|
||||||
border-radius: 8px;
|
|
||||||
}
|
|
||||||
</style>
|
|
@ -1,122 +0,0 @@
|
|||||||
<template>
|
|
||||||
<v-container class="h-screen flex flex-col gap-3 items-center justify-center">
|
|
||||||
<v-card class="w-full max-w-[720px] overflow-auto" :loading="loading">
|
|
||||||
<v-card-text class="card-grid pa-9">
|
|
||||||
<div>
|
|
||||||
<v-avatar color="accent" icon="mdi-lock-reset" size="large" class="card-rounded mb-2" />
|
|
||||||
<h1 class="text-2xl">Reset password</h1>
|
|
||||||
<p>Reset password to get back access of your account.</p>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<v-window :touch="false" :model-value="panel" class="pa-2 mx-[-0.5rem]">
|
|
||||||
<v-window-item value="confirm">
|
|
||||||
<div class="flex items-center">
|
|
||||||
<v-form class="flex-grow-1" @submit.prevent="confirm">
|
|
||||||
<v-text-field
|
|
||||||
label="New Password"
|
|
||||||
type="password"
|
|
||||||
autocomplete="new-password"
|
|
||||||
variant="solo"
|
|
||||||
density="comfortable"
|
|
||||||
:disabled="loading"
|
|
||||||
v-model="newPassword"
|
|
||||||
/>
|
|
||||||
|
|
||||||
<v-expand-transition>
|
|
||||||
<v-alert v-show="error" variant="tonal" type="error" class="text-xs mb-3">
|
|
||||||
Something went wrong... {{ error }}
|
|
||||||
</v-alert>
|
|
||||||
</v-expand-transition>
|
|
||||||
|
|
||||||
<div class="flex justify-end">
|
|
||||||
<v-btn
|
|
||||||
type="submit"
|
|
||||||
variant="text"
|
|
||||||
color="primary"
|
|
||||||
class="justify-self-end"
|
|
||||||
append-icon="mdi-arrow-right"
|
|
||||||
:disabled="loading"
|
|
||||||
>
|
|
||||||
Next
|
|
||||||
</v-btn>
|
|
||||||
</div>
|
|
||||||
</v-form>
|
|
||||||
</div>
|
|
||||||
</v-window-item>
|
|
||||||
<v-window-item value="callback">
|
|
||||||
<div>
|
|
||||||
<v-icon icon="mdi-fire" size="32" color="grey-darken-3" class="mb-3" />
|
|
||||||
|
|
||||||
<h1 class="font-bold text-xl">Applied</h1>
|
|
||||||
<p>The password of your account has updated successfully.</p>
|
|
||||||
|
|
||||||
<p class="mt-3">Now you can continue to use Solarpass, we will redirect you to sign-in soon.</p>
|
|
||||||
</div>
|
|
||||||
</v-window-item>
|
|
||||||
</v-window>
|
|
||||||
</v-card-text>
|
|
||||||
</v-card>
|
|
||||||
|
|
||||||
<copyright />
|
|
||||||
</v-container>
|
|
||||||
</template>
|
|
||||||
|
|
||||||
<script setup lang="ts">
|
|
||||||
import { ref } from "vue"
|
|
||||||
import { useRoute, useRouter } from "vue-router"
|
|
||||||
import { request } from "@/scripts/request"
|
|
||||||
import Copyright from "@/components/Copyright.vue"
|
|
||||||
|
|
||||||
const route = useRoute()
|
|
||||||
const router = useRouter()
|
|
||||||
|
|
||||||
const error = ref<string | null>(null)
|
|
||||||
|
|
||||||
const loading = ref(false)
|
|
||||||
|
|
||||||
const panel = ref("confirm")
|
|
||||||
|
|
||||||
const newPassword = ref("")
|
|
||||||
|
|
||||||
async function confirm() {
|
|
||||||
if (!route.query["code"]) {
|
|
||||||
error.value = "code was not exists"
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
const res = await request("/api/users/me/password-reset", {
|
|
||||||
method: "PATCH",
|
|
||||||
headers: { "Content-Type": "application/json" },
|
|
||||||
body: JSON.stringify({
|
|
||||||
code: route.query["code"],
|
|
||||||
new_password: newPassword.value,
|
|
||||||
}),
|
|
||||||
})
|
|
||||||
if (res.status !== 200) {
|
|
||||||
error.value = await res.text()
|
|
||||||
} else {
|
|
||||||
loading.value = true
|
|
||||||
panel.value = "callback"
|
|
||||||
await router.push({ name: "auth.sign-in" })
|
|
||||||
}
|
|
||||||
loading.value = false
|
|
||||||
}
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<style scoped>
|
|
||||||
.card-grid {
|
|
||||||
display: grid;
|
|
||||||
grid-template-columns: 1fr 1fr;
|
|
||||||
gap: 1rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
@media (max-width: 768px) {
|
|
||||||
.card-grid {
|
|
||||||
grid-template-columns: 1fr;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
.card-rounded {
|
|
||||||
border-radius: 8px;
|
|
||||||
}
|
|
||||||
</style>
|
|
@ -1,157 +0,0 @@
|
|||||||
<template>
|
|
||||||
<div>
|
|
||||||
<go-use-solian class="mb-3" />
|
|
||||||
|
|
||||||
<v-card title="Information" prepend-icon="mdi-face-man-profile" :loading="loading">
|
|
||||||
<template #text>
|
|
||||||
<v-form class="mt-1" @submit.prevent="submit">
|
|
||||||
<v-row dense>
|
|
||||||
<v-col :xs="12" :md="6">
|
|
||||||
<v-text-field readonly hide-details label="Username" density="comfortable" variant="outlined"
|
|
||||||
v-model="data.name" />
|
|
||||||
</v-col>
|
|
||||||
<v-col :xs="12" :md="6">
|
|
||||||
<v-text-field hide-details label="Nickname" density="comfortable" variant="outlined"
|
|
||||||
v-model="data.nick" />
|
|
||||||
</v-col>
|
|
||||||
<v-col :cols="12">
|
|
||||||
<v-textarea hide-details label="Description" density="comfortable" variant="outlined"
|
|
||||||
v-model="data.description" />
|
|
||||||
</v-col>
|
|
||||||
<v-col :xs="12" :md="6" :lg="4">
|
|
||||||
<v-text-field hide-details label="First Name" density="comfortable" variant="outlined"
|
|
||||||
v-model="data.first_name" />
|
|
||||||
</v-col>
|
|
||||||
<v-col :xs="12" :md="6" :lg="4">
|
|
||||||
<v-text-field hide-details label="Last Name" density="comfortable" variant="outlined"
|
|
||||||
v-model="data.last_name" />
|
|
||||||
</v-col>
|
|
||||||
<v-col :xs="12" :lg="4">
|
|
||||||
<v-text-field hide-details label="Birthday" density="comfortable" variant="outlined" type="datetime-local"
|
|
||||||
v-model="data.birthday" />
|
|
||||||
</v-col>
|
|
||||||
</v-row>
|
|
||||||
|
|
||||||
<v-btn type="submit" class="mt-2" variant="text" prepend-icon="mdi-content-save" :disabled="loading">
|
|
||||||
Apply Changes
|
|
||||||
</v-btn>
|
|
||||||
</v-form>
|
|
||||||
</template>
|
|
||||||
</v-card>
|
|
||||||
|
|
||||||
<v-snackbar v-model="done" :timeout="3000"> Your personal information has been updated. </v-snackbar>
|
|
||||||
|
|
||||||
<!-- @vue-ignore -->
|
|
||||||
<v-snackbar v-model="error" :timeout="5000">Something went wrong... {{ error }}</v-snackbar>
|
|
||||||
</div>
|
|
||||||
</template>
|
|
||||||
|
|
||||||
<script setup lang="ts">
|
|
||||||
import { ref, watch } from "vue"
|
|
||||||
import { useUserinfo, getAtk } from "@/stores/userinfo"
|
|
||||||
import { request } from "@/scripts/request"
|
|
||||||
import GoUseSolian from "@/components/GoUseSolian.vue"
|
|
||||||
|
|
||||||
const id = useUserinfo()
|
|
||||||
|
|
||||||
const error = ref<string | null>(null)
|
|
||||||
const done = ref(false)
|
|
||||||
const loading = ref(false)
|
|
||||||
|
|
||||||
const data = ref<any>({})
|
|
||||||
const avatar = ref<any>(null)
|
|
||||||
const banner = ref<any>(null)
|
|
||||||
|
|
||||||
watch(
|
|
||||||
id,
|
|
||||||
(val) => {
|
|
||||||
if (val.isReady) {
|
|
||||||
data.value.name = id.userinfo.data.name
|
|
||||||
data.value.nick = id.userinfo.data.nick
|
|
||||||
data.value.description = id.userinfo.data.description
|
|
||||||
data.value.first_name = id.userinfo.data.profile.first_name
|
|
||||||
data.value.last_name = id.userinfo.data.profile.last_name
|
|
||||||
data.value.birthday = id.userinfo.data.profile.birthday
|
|
||||||
|
|
||||||
if (data.value.birthday) data.value.birthday = data.value.birthday.substring(0, 16)
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{ immediate: true, deep: true },
|
|
||||||
)
|
|
||||||
|
|
||||||
async function submit() {
|
|
||||||
const payload = data.value
|
|
||||||
if (payload.birthday) payload.birthday = new Date(payload.birthday).toISOString()
|
|
||||||
|
|
||||||
loading.value = true
|
|
||||||
const res = await request("/api/users/me", {
|
|
||||||
method: "PUT",
|
|
||||||
headers: { "Content-Type": "application/json", Authorization: `Bearer ${getAtk()}` },
|
|
||||||
body: JSON.stringify(payload),
|
|
||||||
})
|
|
||||||
if (res.status !== 200) {
|
|
||||||
error.value = await res.text()
|
|
||||||
} else {
|
|
||||||
await id.readProfiles()
|
|
||||||
done.value = true
|
|
||||||
error.value = null
|
|
||||||
}
|
|
||||||
loading.value = false
|
|
||||||
}
|
|
||||||
|
|
||||||
async function applyAvatar() {
|
|
||||||
if (!avatar.value) return
|
|
||||||
|
|
||||||
if (loading.value) return
|
|
||||||
|
|
||||||
const payload = new FormData()
|
|
||||||
payload.set("avatar", avatar.value[0])
|
|
||||||
|
|
||||||
loading.value = true
|
|
||||||
const res = await request("/api/users/me/avatar", {
|
|
||||||
method: "PUT",
|
|
||||||
headers: { Authorization: `Bearer ${getAtk()}` },
|
|
||||||
body: payload,
|
|
||||||
})
|
|
||||||
if (res.status !== 200) {
|
|
||||||
error.value = await res.text()
|
|
||||||
} else {
|
|
||||||
await id.readProfiles()
|
|
||||||
done.value = true
|
|
||||||
error.value = null
|
|
||||||
avatar.value = null
|
|
||||||
}
|
|
||||||
loading.value = false
|
|
||||||
}
|
|
||||||
|
|
||||||
async function applyBanner() {
|
|
||||||
if (!banner.value) return
|
|
||||||
|
|
||||||
if (loading.value) return
|
|
||||||
|
|
||||||
const payload = new FormData()
|
|
||||||
payload.set("banner", banner.value[0])
|
|
||||||
|
|
||||||
loading.value = true
|
|
||||||
const res = await request("/api/users/me/banner", {
|
|
||||||
method: "PUT",
|
|
||||||
headers: { Authorization: `Bearer ${getAtk()}` },
|
|
||||||
body: payload,
|
|
||||||
})
|
|
||||||
if (res.status !== 200) {
|
|
||||||
error.value = await res.text()
|
|
||||||
} else {
|
|
||||||
await id.readProfiles()
|
|
||||||
done.value = true
|
|
||||||
error.value = null
|
|
||||||
banner.value = null
|
|
||||||
}
|
|
||||||
loading.value = false
|
|
||||||
}
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<style>
|
|
||||||
.rounded-card {
|
|
||||||
border-radius: 8px;
|
|
||||||
}
|
|
||||||
</style>
|
|
@ -1,197 +0,0 @@
|
|||||||
<template>
|
|
||||||
<div>
|
|
||||||
<v-expansion-panels>
|
|
||||||
<v-expansion-panel eager title="Tickets">
|
|
||||||
<template #text>
|
|
||||||
<v-card :loading="reverting.tickets" variant="outlined">
|
|
||||||
<v-data-table-server
|
|
||||||
density="compact"
|
|
||||||
:headers="dataDefinitions.tickets"
|
|
||||||
:items="tickets"
|
|
||||||
:items-length="pagination.tickets.total"
|
|
||||||
:loading="reverting.tickets"
|
|
||||||
v-model:items-per-page="pagination.tickets.pageSize"
|
|
||||||
@update:options="readTickets"
|
|
||||||
item-value="id"
|
|
||||||
>
|
|
||||||
<template v-slot:item="{ item }: { item: any }">
|
|
||||||
<tr>
|
|
||||||
<td>{{ item.id }}</td>
|
|
||||||
<td>{{ item.ip_address }}</td>
|
|
||||||
<td>
|
|
||||||
<v-tooltip :text="item.user_agent" location="top">
|
|
||||||
<template #activator="{ props }">
|
|
||||||
<div v-bind="props" class="text-ellipsis whitespace-nowrap overflow-hidden max-w-[280px]">
|
|
||||||
{{ item.user_agent }}
|
|
||||||
</div>
|
|
||||||
</template>
|
|
||||||
</v-tooltip>
|
|
||||||
</td>
|
|
||||||
<td>{{ new Date(item.created_at).toLocaleString() }}</td>
|
|
||||||
<td>
|
|
||||||
<v-tooltip text="Sign Out">
|
|
||||||
<template #activator="{ props }">
|
|
||||||
<v-btn
|
|
||||||
v-bind="props"
|
|
||||||
variant="text"
|
|
||||||
size="x-small"
|
|
||||||
color="error"
|
|
||||||
icon="mdi-logout-variant"
|
|
||||||
@click="killTicket(item)"
|
|
||||||
/>
|
|
||||||
</template>
|
|
||||||
</v-tooltip>
|
|
||||||
</td>
|
|
||||||
</tr>
|
|
||||||
</template>
|
|
||||||
</v-data-table-server>
|
|
||||||
</v-card>
|
|
||||||
</template>
|
|
||||||
</v-expansion-panel>
|
|
||||||
|
|
||||||
<v-expansion-panel eager title="Events">
|
|
||||||
<template #text>
|
|
||||||
<v-card :loading="reverting.events" variant="outlined">
|
|
||||||
<v-data-table-server
|
|
||||||
density="compact"
|
|
||||||
:headers="dataDefinitions.events"
|
|
||||||
:items="events"
|
|
||||||
:items-length="pagination.events.total"
|
|
||||||
:loading="reverting.events"
|
|
||||||
v-model:items-per-page="pagination.events.pageSize"
|
|
||||||
@update:options="readEvents"
|
|
||||||
item-value="id"
|
|
||||||
>
|
|
||||||
<template v-slot:item="{ item }: { item: any }">
|
|
||||||
<tr>
|
|
||||||
<td>{{ item.id }}</td>
|
|
||||||
<td>{{ item.type }}</td>
|
|
||||||
<td>{{ item.target }}</td>
|
|
||||||
<td>{{ item.ip_address }}</td>
|
|
||||||
<td>
|
|
||||||
<v-tooltip :text="item.user_agent" location="top">
|
|
||||||
<template #activator="{ props }">
|
|
||||||
<div v-bind="props" class="text-ellipsis whitespace-nowrap overflow-hidden max-w-[180px]">
|
|
||||||
{{ item.user_agent }}
|
|
||||||
</div>
|
|
||||||
</template>
|
|
||||||
</v-tooltip>
|
|
||||||
</td>
|
|
||||||
<td>{{ new Date(item.created_at).toLocaleString() }}</td>
|
|
||||||
</tr>
|
|
||||||
</template>
|
|
||||||
</v-data-table-server>
|
|
||||||
</v-card>
|
|
||||||
</template>
|
|
||||||
</v-expansion-panel>
|
|
||||||
</v-expansion-panels>
|
|
||||||
</div>
|
|
||||||
</template>
|
|
||||||
|
|
||||||
<script setup lang="ts">
|
|
||||||
import { request } from "@/scripts/request"
|
|
||||||
import { getAtk } from "@/stores/userinfo"
|
|
||||||
import { reactive, ref } from "vue"
|
|
||||||
|
|
||||||
const error = ref<string | null>(null)
|
|
||||||
|
|
||||||
const dataDefinitions: { [id: string]: any[] } = {
|
|
||||||
tickets: [
|
|
||||||
{ align: "start", key: "id", title: "ID" },
|
|
||||||
{ align: "start", key: "ip_address", title: "IP Address" },
|
|
||||||
{ align: "start", key: "user_agent", title: "User Agent" },
|
|
||||||
{ align: "start", key: "created_at", title: "Issued At" },
|
|
||||||
{ align: "start", key: "actions", title: "Actions", sortable: false },
|
|
||||||
],
|
|
||||||
events: [
|
|
||||||
{ align: "start", key: "id", title: "ID" },
|
|
||||||
{ align: "start", key: "type", title: "Type" },
|
|
||||||
{ align: "start", key: "target", title: "Affected Object" },
|
|
||||||
{ align: "start", key: "ip_address", title: "IP Address" },
|
|
||||||
{ align: "start", key: "user_agent", title: "User Agent" },
|
|
||||||
{ align: "start", key: "created_at", title: "Performed At" },
|
|
||||||
],
|
|
||||||
}
|
|
||||||
|
|
||||||
const tickets = ref<any>([])
|
|
||||||
const events = ref<any>([])
|
|
||||||
|
|
||||||
const reverting = reactive({ tickets: false, sessions: false, events: false })
|
|
||||||
const pagination = reactive({
|
|
||||||
tickets: { page: 1, pageSize: 5, total: 0 },
|
|
||||||
events: { page: 1, pageSize: 5, total: 0 },
|
|
||||||
})
|
|
||||||
|
|
||||||
async function readTickets({ page, itemsPerPage }: { page?: number; itemsPerPage?: number }) {
|
|
||||||
if (itemsPerPage) pagination.tickets.pageSize = itemsPerPage
|
|
||||||
if (page) pagination.tickets.page = page
|
|
||||||
|
|
||||||
reverting.sessions = true
|
|
||||||
const res = await request(
|
|
||||||
"/api/users/me/tickets?" +
|
|
||||||
new URLSearchParams({
|
|
||||||
take: pagination.tickets.pageSize.toString(),
|
|
||||||
offset: ((pagination.tickets.page - 1) * pagination.tickets.pageSize).toString(),
|
|
||||||
}),
|
|
||||||
{
|
|
||||||
headers: { Authorization: `Bearer ${getAtk()}` },
|
|
||||||
},
|
|
||||||
)
|
|
||||||
if (res.status !== 200) {
|
|
||||||
error.value = await res.text()
|
|
||||||
} else {
|
|
||||||
const data = await res.json()
|
|
||||||
tickets.value = data["data"]
|
|
||||||
pagination.tickets.total = data["count"]
|
|
||||||
}
|
|
||||||
reverting.sessions = false
|
|
||||||
}
|
|
||||||
|
|
||||||
async function readEvents({ page, itemsPerPage }: { page?: number; itemsPerPage?: number }) {
|
|
||||||
if (itemsPerPage) pagination.events.pageSize = itemsPerPage
|
|
||||||
if (page) pagination.events.page = page
|
|
||||||
|
|
||||||
reverting.events = true
|
|
||||||
const res = await request(
|
|
||||||
"/api/users/me/events?" +
|
|
||||||
new URLSearchParams({
|
|
||||||
take: pagination.events.pageSize.toString(),
|
|
||||||
offset: ((pagination.events.page - 1) * pagination.events.pageSize).toString(),
|
|
||||||
}),
|
|
||||||
{
|
|
||||||
headers: { Authorization: `Bearer ${getAtk()}` },
|
|
||||||
},
|
|
||||||
)
|
|
||||||
if (res.status !== 200) {
|
|
||||||
error.value = await res.text()
|
|
||||||
} else {
|
|
||||||
const data = await res.json()
|
|
||||||
events.value = data["data"]
|
|
||||||
pagination.events.total = data["count"]
|
|
||||||
}
|
|
||||||
reverting.events = false
|
|
||||||
}
|
|
||||||
|
|
||||||
Promise.all([readTickets({}), readEvents({})])
|
|
||||||
|
|
||||||
async function killTicket(item: any) {
|
|
||||||
reverting.sessions = true
|
|
||||||
const res = await request(`/api/users/me/tickets/${item.id}`, {
|
|
||||||
method: "DELETE",
|
|
||||||
headers: { Authorization: `Bearer ${getAtk()}` },
|
|
||||||
})
|
|
||||||
if (res.status !== 200) {
|
|
||||||
error.value = await res.text()
|
|
||||||
} else {
|
|
||||||
await readTickets({})
|
|
||||||
error.value = null
|
|
||||||
}
|
|
||||||
reverting.sessions = false
|
|
||||||
}
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<style>
|
|
||||||
.rounded-card {
|
|
||||||
border-radius: 8px;
|
|
||||||
}
|
|
||||||
</style>
|
|
@ -1,14 +0,0 @@
|
|||||||
{
|
|
||||||
"extends": "@vue/tsconfig/tsconfig.dom.json",
|
|
||||||
"include": ["env.d.ts", "src/**/*", "src/**/*.vue"],
|
|
||||||
"exclude": ["src/**/__tests__/*"],
|
|
||||||
"compilerOptions": {
|
|
||||||
"composite": true,
|
|
||||||
"tsBuildInfoFile": "./node_modules/.tmp/tsconfig.app.tsbuildinfo",
|
|
||||||
|
|
||||||
"baseUrl": ".",
|
|
||||||
"paths": {
|
|
||||||
"@/*": ["./src/*"]
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,11 +0,0 @@
|
|||||||
{
|
|
||||||
"files": [],
|
|
||||||
"references": [
|
|
||||||
{
|
|
||||||
"path": "./tsconfig.node.json"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"path": "./tsconfig.app.json"
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
@ -1,13 +0,0 @@
|
|||||||
{
|
|
||||||
"extends": "@tsconfig/node20/tsconfig.json",
|
|
||||||
"include": ["vite.config.*", "vitest.config.*", "cypress.config.*", "nightwatch.conf.*", "playwright.config.*"],
|
|
||||||
"compilerOptions": {
|
|
||||||
"composite": true,
|
|
||||||
"noEmit": true,
|
|
||||||
"tsBuildInfoFile": "./node_modules/.tmp/tsconfig.node.tsbuildinfo",
|
|
||||||
|
|
||||||
"module": "ESNext",
|
|
||||||
"moduleResolution": "Bundler",
|
|
||||||
"types": ["node"]
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,5 +0,0 @@
|
|||||||
import { defineConfig, presetAttributify, presetTypography, presetUno } from "unocss"
|
|
||||||
|
|
||||||
export default defineConfig({
|
|
||||||
presets: [presetAttributify(), presetTypography(), presetUno({ preflight: false })],
|
|
||||||
})
|
|
@ -1,27 +0,0 @@
|
|||||||
import { fileURLToPath, URL } from "node:url";
|
|
||||||
|
|
||||||
import { defineConfig } from "vite";
|
|
||||||
import vue from "@vitejs/plugin-vue";
|
|
||||||
import vueJsx from "@vitejs/plugin-vue-jsx";
|
|
||||||
import unocss from "unocss/vite";
|
|
||||||
|
|
||||||
// https://vitejs.dev/config/
|
|
||||||
export default defineConfig({
|
|
||||||
plugins: [vue(), vueJsx(), unocss()],
|
|
||||||
resolve: {
|
|
||||||
alias: {
|
|
||||||
"@": fileURLToPath(new URL("./src", import.meta.url))
|
|
||||||
}
|
|
||||||
},
|
|
||||||
server: {
|
|
||||||
proxy: {
|
|
||||||
"/api/ws": {
|
|
||||||
target: "ws://localhost:8444",
|
|
||||||
ws: true
|
|
||||||
},
|
|
||||||
|
|
||||||
"/api": "http://localhost:8444",
|
|
||||||
"/.well-known": "http://localhost:8444"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
Loading…
Reference in New Issue
Block a user