🔀 Merge pull request '♻️ Use Capital (the new portal website) instead of embed frontend)' (#6) from refactor/use-capital-as-frontend into master
Reviewed-on: Hydrogen/Passport#6
This commit is contained in:
		
							
								
								
									
										2
									
								
								.idea/dataSources.local.xml
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										2
									
								
								.idea/dataSources.local.xml
									
									
									
										generated
									
									
									
								
							@@ -1,6 +1,6 @@
 | 
			
		||||
<?xml version="1.0" encoding="UTF-8"?>
 | 
			
		||||
<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">
 | 
			
		||||
      <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>
 | 
			
		||||
 
 | 
			
		||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										66
									
								
								.idea/workspace.xml
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										66
									
								
								.idea/workspace.xml
									
									
									
										generated
									
									
									
								
							@@ -4,7 +4,13 @@
 | 
			
		||||
    <option name="autoReloadType" value="ALL" />
 | 
			
		||||
  </component>
 | 
			
		||||
  <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=":wastebasket: Clean up code">
 | 
			
		||||
      <change beforePath="$PROJECT_DIR$/.idea/workspace.xml" beforeDir="false" afterPath="$PROJECT_DIR$/.idea/workspace.xml" afterDir="false" />
 | 
			
		||||
      <change beforePath="$PROJECT_DIR$/pkg/internal/server/api/well_known_api.go" beforeDir="false" afterPath="$PROJECT_DIR$/pkg/internal/server/api/well_known_api.go" afterDir="false" />
 | 
			
		||||
      <change beforePath="$PROJECT_DIR$/pkg/internal/services/jwt.go" beforeDir="false" afterPath="$PROJECT_DIR$/pkg/internal/services/jwt.go" afterDir="false" />
 | 
			
		||||
      <change beforePath="$PROJECT_DIR$/pkg/internal/services/ticket.go" beforeDir="false" afterPath="$PROJECT_DIR$/pkg/internal/services/ticket.go" afterDir="false" />
 | 
			
		||||
      <change beforePath="$PROJECT_DIR$/settings.toml" beforeDir="false" afterPath="$PROJECT_DIR$/settings.toml" afterDir="false" />
 | 
			
		||||
    </list>
 | 
			
		||||
    <option name="SHOW_DIALOG" value="false" />
 | 
			
		||||
    <option name="HIGHLIGHT_CONFLICTS" value="true" />
 | 
			
		||||
    <option name="HIGHLIGHT_NON_ACTIVE_CHANGELIST" value="false" />
 | 
			
		||||
@@ -38,34 +44,34 @@
 | 
			
		||||
    <option name="hideEmptyMiddlePackages" value="true" />
 | 
			
		||||
    <option name="showLibraryContents" value="true" />
 | 
			
		||||
  </component>
 | 
			
		||||
  <component name="PropertiesComponent">{
 | 
			
		||||
  "keyToString": {
 | 
			
		||||
    "DefaultGoTemplateProperty": "Go File",
 | 
			
		||||
    "Go Build.Backend.executor": "Run",
 | 
			
		||||
    "Go 构建.Backend.executor": "Run",
 | 
			
		||||
    "RunOnceActivity.ShowReadmeOnStart": "true",
 | 
			
		||||
    "RunOnceActivity.go.formatter.settings.were.checked": "true",
 | 
			
		||||
    "RunOnceActivity.go.migrated.go.modules.settings": "true",
 | 
			
		||||
    "RunOnceActivity.go.modules.automatic.dependencies.download": "true",
 | 
			
		||||
    "RunOnceActivity.go.modules.go.list.on.any.changes.was.set": "true",
 | 
			
		||||
    "git-widget-placeholder": "master",
 | 
			
		||||
    "go.import.settings.migrated": "true",
 | 
			
		||||
    "go.sdk.automatically.set": "true",
 | 
			
		||||
    "last_opened_file_path": "/Users/littlesheep/Documents/Projects/Hydrogen/Passport/web/src/components/admin",
 | 
			
		||||
    "node.js.detected.package.eslint": "true",
 | 
			
		||||
    "node.js.selected.package.eslint": "(autodetect)",
 | 
			
		||||
    "nodejs_package_manager_path": "npm",
 | 
			
		||||
    "run.code.analysis.last.selected.profile": "pProject Default",
 | 
			
		||||
    "settings.editor.selected.configurable": "preferences.pluginManager",
 | 
			
		||||
    "ts.external.directory.path": "/Users/littlesheep/Documents/Projects/Hydrogen/Passport/web/node_modules/typescript/lib",
 | 
			
		||||
    "vue.rearranger.settings.migration": "true"
 | 
			
		||||
  <component name="PropertiesComponent"><![CDATA[{
 | 
			
		||||
  "keyToString": {
 | 
			
		||||
    "DefaultGoTemplateProperty": "Go File",
 | 
			
		||||
    "Go Build.Backend.executor": "Run",
 | 
			
		||||
    "Go 构建.Backend.executor": "Run",
 | 
			
		||||
    "RunOnceActivity.ShowReadmeOnStart": "true",
 | 
			
		||||
    "RunOnceActivity.go.formatter.settings.were.checked": "true",
 | 
			
		||||
    "RunOnceActivity.go.migrated.go.modules.settings": "true",
 | 
			
		||||
    "RunOnceActivity.go.modules.automatic.dependencies.download": "true",
 | 
			
		||||
    "RunOnceActivity.go.modules.go.list.on.any.changes.was.set": "true",
 | 
			
		||||
    "git-widget-placeholder": "refactor/use-capital-as-frontend",
 | 
			
		||||
    "go.import.settings.migrated": "true",
 | 
			
		||||
    "go.sdk.automatically.set": "true",
 | 
			
		||||
    "last_opened_file_path": "/Users/littlesheep/Documents/Projects/Hydrogen/Passport/web/src/components/admin",
 | 
			
		||||
    "node.js.detected.package.eslint": "true",
 | 
			
		||||
    "node.js.selected.package.eslint": "(autodetect)",
 | 
			
		||||
    "nodejs_package_manager_path": "npm",
 | 
			
		||||
    "run.code.analysis.last.selected.profile": "pProject Default",
 | 
			
		||||
    "settings.editor.selected.configurable": "preferences.pluginManager",
 | 
			
		||||
    "ts.external.directory.path": "/Users/littlesheep/Documents/Projects/Hydrogen/Passport/web/node_modules/typescript/lib",
 | 
			
		||||
    "vue.rearranger.settings.migration": "true"
 | 
			
		||||
  },
 | 
			
		||||
  "keyToStringList": {
 | 
			
		||||
    "DatabaseDriversLRU": [
 | 
			
		||||
      "postgresql"
 | 
			
		||||
  "keyToStringList": {
 | 
			
		||||
    "DatabaseDriversLRU": [
 | 
			
		||||
      "postgresql"
 | 
			
		||||
    ]
 | 
			
		||||
  }
 | 
			
		||||
}</component>
 | 
			
		||||
}]]></component>
 | 
			
		||||
  <component name="RecentsManager">
 | 
			
		||||
    <key name="CopyFile.RECENT_KEYS">
 | 
			
		||||
      <recent name="$PROJECT_DIR$/web/src/components/admin" />
 | 
			
		||||
@@ -109,8 +115,8 @@
 | 
			
		||||
  <component name="SharedIndexes">
 | 
			
		||||
    <attachedChunks>
 | 
			
		||||
      <set>
 | 
			
		||||
        <option value="bundled-gosdk-33c477a475b1-e0158606a674-org.jetbrains.plugins.go.sharedIndexes.bundled-GO-241.18034.61" />
 | 
			
		||||
        <option value="bundled-js-predefined-1d06a55b98c1-0b3e54e931b4-JavaScript-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-d6986cc7102b-410509235cf1-JavaScript-GO-242.20224.306" />
 | 
			
		||||
      </set>
 | 
			
		||||
    </attachedChunks>
 | 
			
		||||
  </component>
 | 
			
		||||
@@ -150,7 +156,6 @@
 | 
			
		||||
    </option>
 | 
			
		||||
  </component>
 | 
			
		||||
  <component name="VcsManagerConfiguration">
 | 
			
		||||
    <MESSAGE value=":sparkles: Reset password APIs" />
 | 
			
		||||
    <MESSAGE value=":sparkles: Password reset & user lookup API" />
 | 
			
		||||
    <MESSAGE value=":recycle: Optimized the initial permission system" />
 | 
			
		||||
    <MESSAGE value=":zap: Optimized audit, event logging system
:sparkles: Audit logs
:sparkles: Admin edit user permissions" />
 | 
			
		||||
@@ -175,7 +180,8 @@
 | 
			
		||||
    <MESSAGE value=":sparkles: Account groups" />
 | 
			
		||||
    <MESSAGE value=":sparkles: Default user group" />
 | 
			
		||||
    <MESSAGE value=":bug: Fix permissions in groups" />
 | 
			
		||||
    <option name="LAST_COMMIT_MESSAGE" value=":bug: Fix permissions in groups" />
 | 
			
		||||
    <MESSAGE value=":wastebasket: Clean up code" />
 | 
			
		||||
    <option name="LAST_COMMIT_MESSAGE" value=":wastebasket: Clean up code" />
 | 
			
		||||
  </component>
 | 
			
		||||
  <component name="VgoProject">
 | 
			
		||||
    <settings-migrated>true</settings-migrated>
 | 
			
		||||
 
 | 
			
		||||
@@ -1,23 +1,14 @@
 | 
			
		||||
# Building Backend
 | 
			
		||||
FROM golang:alpine as passport-server
 | 
			
		||||
 | 
			
		||||
RUN apk add nodejs npm
 | 
			
		||||
 | 
			
		||||
WORKDIR /source
 | 
			
		||||
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
 | 
			
		||||
 | 
			
		||||
# Runtime
 | 
			
		||||
FROM golang:alpine
 | 
			
		||||
 | 
			
		||||
COPY --from=passport-server /dist /passport/server
 | 
			
		||||
COPY --from=passport-server /source/web/dist /passport/web
 | 
			
		||||
 | 
			
		||||
EXPOSE 8444
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										2
									
								
								go.mod
									
									
									
									
									
								
							
							
						
						
									
										2
									
								
								go.mod
									
									
									
									
									
								
							@@ -5,7 +5,7 @@ go 1.21.6
 | 
			
		||||
toolchain go1.22.1
 | 
			
		||||
 | 
			
		||||
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/gofiber/fiber/v2 v2.52.4
 | 
			
		||||
	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/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/go.mod h1:CxXYINrC8qIiEnFrOxCa7Jy5BFHlXnUU2pbicEuybxQ=
 | 
			
		||||
github.com/DataDog/datadog-go v3.2.0+incompatible/go.mod h1:LButxg5PwREeZtORoXG3tL4fMGNddJ+vMq1mwgfaqoQ=
 | 
			
		||||
 
 | 
			
		||||
@@ -12,7 +12,7 @@ func getOidcConfiguration(c *fiber.Ctx) error {
 | 
			
		||||
	basepath := fmt.Sprintf("https://%s", domain)
 | 
			
		||||
 | 
			
		||||
	return c.JSON(fiber.Map{
 | 
			
		||||
		"issuer":                                           basepath,
 | 
			
		||||
		"issuer":                                           viper.GetString("security.issuer"),
 | 
			
		||||
		"authorization_endpoint":                           fmt.Sprintf("%s/authorize", basepath),
 | 
			
		||||
		"token_endpoint":                                   fmt.Sprintf("%s/api/auth/token", basepath),
 | 
			
		||||
		"userinfo_endpoint":                                fmt.Sprintf("%s/api/users/me", basepath),
 | 
			
		||||
 
 | 
			
		||||
@@ -20,7 +20,7 @@ type PayloadClaims struct {
 | 
			
		||||
	Nick  string `json:"preferred_username,omitempty"`
 | 
			
		||||
	Email string `json:"email,omitempty"`
 | 
			
		||||
 | 
			
		||||
	// Additonal Stuff
 | 
			
		||||
	// Additional Stuff
 | 
			
		||||
	AuthorizedParties string `json:"azp,omitempty"`
 | 
			
		||||
	Nonce             string `json:"nonce,omitempty"`
 | 
			
		||||
	Type              string `json:"typ"`
 | 
			
		||||
@@ -44,7 +44,7 @@ func EncodeJwt(id string, typ, sub, sed string, nonce *string, aud []string, exp
 | 
			
		||||
		RegisteredClaims: jwt.RegisteredClaims{
 | 
			
		||||
			Subject:   sub,
 | 
			
		||||
			Audience:  aud,
 | 
			
		||||
			Issuer:    fmt.Sprintf("https://%s", viper.GetString("domain")),
 | 
			
		||||
			Issuer:    viper.GetString("security.issuer"),
 | 
			
		||||
			ExpiresAt: jwt.NewNumericDate(exp),
 | 
			
		||||
			NotBefore: jwt.NewNumericDate(time.Now()),
 | 
			
		||||
			IssuedAt:  jwt.NewNumericDate(time.Now()),
 | 
			
		||||
 
 | 
			
		||||
@@ -11,7 +11,7 @@ import (
 | 
			
		||||
	"github.com/samber/lo"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
const InternalTokenAudience = "passport"
 | 
			
		||||
const InternalTokenAudience = "solar-network"
 | 
			
		||||
 | 
			
		||||
func DetectRisk(user models.Account, ip, ua string) bool {
 | 
			
		||||
	var clue int64
 | 
			
		||||
 
 | 
			
		||||
@@ -1,12 +1,12 @@
 | 
			
		||||
id = "passport01"
 | 
			
		||||
name = "Solarpass"
 | 
			
		||||
 | 
			
		||||
frontend_app = "web/dist"
 | 
			
		||||
 | 
			
		||||
bind = "0.0.0.0:8444"
 | 
			
		||||
grpc_bind = "0.0.0.0:7444"
 | 
			
		||||
domain = "localhost"
 | 
			
		||||
 | 
			
		||||
domain = "id.solsynth.dev"
 | 
			
		||||
 | 
			
		||||
content_endpoint = "https://usercontent.solsynth.dev"
 | 
			
		||||
 | 
			
		||||
default_user_group = 1
 | 
			
		||||
@@ -21,6 +21,7 @@ print_routes = false
 | 
			
		||||
addr = "127.0.0.1:7442"
 | 
			
		||||
 | 
			
		||||
[security]
 | 
			
		||||
issuer = "https://solsynth.dev"
 | 
			
		||||
cookie_domain = "localhost"
 | 
			
		||||
cookie_samesite = "Lax"
 | 
			
		||||
access_token_duration = 300
 | 
			
		||||
 
 | 
			
		||||
@@ -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"
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
});
 | 
			
		||||
		Reference in New Issue
	
	Block a user