From a7c48d900cb5d2ab53954e0cf91a934d2f7242a4 Mon Sep 17 00:00:00 2001 From: Javier Caceres <81489497+srnovus@users.noreply.github.com> Date: Wed, 30 Oct 2024 13:48:29 -0600 Subject: [PATCH 1/5] nvus rama (#23) * add ci * add github yml * update ci * update ci * update error ci * update * resolver issue #6 --- .github/fedired/test.yml | 16 ++++++++ .github/labeler.yml | 15 +++++++ .github/workflows/ci.yml | 39 +++++++++++++++++++ .github/workflows/labeler.yml | 16 ++++++++ .../src/remote/activitypub/resolver.ts | 11 +++--- 5 files changed, 92 insertions(+), 5 deletions(-) create mode 100644 .github/fedired/test.yml create mode 100644 .github/labeler.yml create mode 100644 .github/workflows/ci.yml create mode 100644 .github/workflows/labeler.yml diff --git a/.github/fedired/test.yml b/.github/fedired/test.yml new file mode 100644 index 0000000..1b807bc --- /dev/null +++ b/.github/fedired/test.yml @@ -0,0 +1,16 @@ +url: 'http://fedired.local' + +setupPassword: example_password_please_change_this_or_you_will_get_hacked + +port: 61812 + +db: + host: 127.0.0.1 + port: 54312 + db: test-fedired + user: postgres + pass: '' +redis: + host: 127.0.0.1 + port: 56312 +id: aidx \ No newline at end of file diff --git a/.github/labeler.yml b/.github/labeler.yml new file mode 100644 index 0000000..27947ea --- /dev/null +++ b/.github/labeler.yml @@ -0,0 +1,15 @@ +# .github/labeler.yml + +version: 2 + +pull_request: + 'packages/backend/**': + - backend + 'packages/backend-rs/**': + - backend-rust + 'packages/client/**': + - client + 'packages/sw/**': + - service-worker + 'packages/fedired-js/**': + - frontend diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml new file mode 100644 index 0000000..8d604cb --- /dev/null +++ b/.github/workflows/ci.yml @@ -0,0 +1,39 @@ +name: Test (production install and build) + +on: + push: + branches: + - master + - develop + pull_request: + +env: + NODE_ENV: production + +jobs: + production: + runs-on: ubuntu-latest + + strategy: + matrix: + node-version: [20.16.0] + + steps: + - uses: actions/checkout@v4.1.1 + with: + submodules: true + - name: Install pnpm + uses: pnpm/action-setup@v4 + - name: Use Node.js ${{ matrix.node-version }} + uses: actions/setup-node@v4.0.4 + with: + node-version: ${{ matrix.node-version }} + cache: 'pnpm' + - run: corepack enable + - run: pnpm i --no-frozen-lockfile # Cambiado aquí + - name: Check pnpm-lock.yaml + run: git diff --exit-code pnpm-lock.yaml + - name: Copy Configure + run: cp .github/fedired/test.yml .config/default.yml + - name: Build + run: pnpm build diff --git a/.github/workflows/labeler.yml b/.github/workflows/labeler.yml new file mode 100644 index 0000000..cab313b --- /dev/null +++ b/.github/workflows/labeler.yml @@ -0,0 +1,16 @@ +name: "Pull Request Labeler" +on: + pull_request_target: + branches-ignore: + - 'l10n_develop' + +jobs: + triage: + permissions: + contents: read + pull-requests: write + runs-on: ubuntu-latest + steps: + - uses: actions/labeler@v5 + with: + repo-token: "${{ secrets.GITHUB_TOKEN }}" \ No newline at end of file diff --git a/packages/backend/src/remote/activitypub/resolver.ts b/packages/backend/src/remote/activitypub/resolver.ts index 7e6e223..5d0cacd 100644 --- a/packages/backend/src/remote/activitypub/resolver.ts +++ b/packages/backend/src/remote/activitypub/resolver.ts @@ -1,4 +1,5 @@ import { config } from "@/config.js"; +const allowedContexts = ["https://www.w3.org/ns/activitystreams"]; import type { ILocalUser } from "@/models/entities/user.js"; import { extractHost, @@ -125,10 +126,10 @@ export default class Resolver { if ( object == null || (Array.isArray(object["@context"]) - ? !(object["@context"] as unknown[]).includes( - "https://www.w3.org/ns/activitystreams", - ) - : object["@context"] !== "https://www.w3.org/ns/activitystreams") + ? !((object["@context"] as unknown[]).every( + (ctx) => allowedContexts.includes(ctx) + )) + : !allowedContexts.includes(object["@context"])) ) { throw new Error("invalid response"); } @@ -219,4 +220,4 @@ export default class Resolver { throw new Error(`resolveLocal: type ${parsed.type} unhandled`); } } -} +} \ No newline at end of file From be63ef1b11132ccff9219a2cfc4228737c4148ef Mon Sep 17 00:00:00 2001 From: Javier Caceres <81489497+srnovus@users.noreply.github.com> Date: Wed, 30 Oct 2024 15:23:27 -0600 Subject: [PATCH 2/5] Create node.js.yml Signed-off-by: Javier Caceres <81489497+srnovus@users.noreply.github.com> --- .github/workflows/node.js.yml | 31 +++++++++++++++++++++++++++++++ 1 file changed, 31 insertions(+) create mode 100644 .github/workflows/node.js.yml diff --git a/.github/workflows/node.js.yml b/.github/workflows/node.js.yml new file mode 100644 index 0000000..2284b93 --- /dev/null +++ b/.github/workflows/node.js.yml @@ -0,0 +1,31 @@ +# This workflow will do a clean installation of node dependencies, cache/restore them, build the source code and run tests across different versions of node +# For more information see: https://docs.github.com/en/actions/automating-builds-and-tests/building-and-testing-nodejs + +name: Node.js CI + +on: + push: + branches: [ "main" ] + pull_request: + branches: [ "main" ] + +jobs: + build: + + runs-on: ubuntu-latest + + strategy: + matrix: + node-version: [18.x, 20.x, 22.x] + # See supported Node.js release schedule at https://nodejs.org/en/about/releases/ + + steps: + - uses: actions/checkout@v4 + - name: Use Node.js ${{ matrix.node-version }} + uses: actions/setup-node@v4 + with: + node-version: ${{ matrix.node-version }} + cache: 'npm' + - run: npm ci + - run: npm run build --if-present + - run: npm test From 18af9ef5ebf1a5d1ef1824e043ceb26e53e23b09 Mon Sep 17 00:00:00 2001 From: Javier Caceres <81489497+srnovus@users.noreply.github.com> Date: Wed, 30 Oct 2024 18:40:44 -0600 Subject: [PATCH 3/5] Nvus (#24) * add ci * add github yml * update ci * update ci * update error ci * update * resolver issue #6 * ci * udpate ci * update ci * a * add github integration and update mknote * update --- .github/workflows/npm-publish.yml | 33 -- packages/backend/src/models/entities/meta.ts | 17 + .../src/server/api/endpoints/admin/meta.ts | 5 + .../server/api/endpoints/admin/update-meta.ts | 3 + .../backend/src/server/api/endpoints/meta.ts | 5 + .../backend/src/server/api/service/github.ts | 295 ++++++++++++++++++ packages/client/src/components/MkSignin.vue | 12 + .../src/components/note/MkNoteHeader.vue | 39 +-- .../src/pages/admin/integrations.github.vue | 69 ++++ .../client/src/pages/settings/integration.vue | 67 ++++ packages/fedired-js/src/entities.ts | 1 + 11 files changed, 488 insertions(+), 58 deletions(-) delete mode 100644 .github/workflows/npm-publish.yml create mode 100644 packages/backend/src/server/api/service/github.ts create mode 100644 packages/client/src/pages/admin/integrations.github.vue create mode 100644 packages/client/src/pages/settings/integration.vue diff --git a/.github/workflows/npm-publish.yml b/.github/workflows/npm-publish.yml deleted file mode 100644 index 2a4766d..0000000 --- a/.github/workflows/npm-publish.yml +++ /dev/null @@ -1,33 +0,0 @@ -# This workflow will run tests using node and then publish a package to GitHub Packages when a release is created -# For more information see: https://docs.github.com/en/actions/publishing-packages/publishing-nodejs-packages - -name: Node.js Package - -on: - release: - types: [created] - -jobs: - build: - runs-on: ubuntu-latest - steps: - - uses: actions/checkout@v4 - - uses: actions/setup-node@v4 - with: - node-version: 20 - - run: npm ci - - run: npm test - - publish-npm: - needs: build - runs-on: ubuntu-latest - steps: - - uses: actions/checkout@v4 - - uses: actions/setup-node@v4 - with: - node-version: 20 - registry-url: https://registry.npmjs.org/ - - run: npm ci - - run: npm publish - env: - NODE_AUTH_TOKEN: ${{secrets.npm_token}} diff --git a/packages/backend/src/models/entities/meta.ts b/packages/backend/src/models/entities/meta.ts index b6a32e3..73a495f 100644 --- a/packages/backend/src/models/entities/meta.ts +++ b/packages/backend/src/models/entities/meta.ts @@ -348,6 +348,23 @@ export class Meta { length: 128, nullable: true, }) + public enableGithubIntegration: boolean; + + @Column("varchar", { + length: 128, + nullable: true, + }) + public githubClientId: string | null; + + @Column("varchar", { + length: 128, + nullable: true, + }) + public githubClientSecret: string | null; + + @Column("boolean", { + default: false, + }) public deeplAuthKey: string | null; @Column("boolean", { diff --git a/packages/backend/src/server/api/endpoints/admin/meta.ts b/packages/backend/src/server/api/endpoints/admin/meta.ts index 674262c..6eaca6c 100644 --- a/packages/backend/src/server/api/endpoints/admin/meta.ts +++ b/packages/backend/src/server/api/endpoints/admin/meta.ts @@ -178,6 +178,11 @@ export const meta = { optional: false, nullable: false, }, + enableGithubIntegration: { + type: "boolean", + optional: false, + nullable: false, + }, enableServiceWorker: { type: "boolean", optional: false, diff --git a/packages/backend/src/server/api/endpoints/admin/update-meta.ts b/packages/backend/src/server/api/endpoints/admin/update-meta.ts index 210b5d2..01f1fe2 100644 --- a/packages/backend/src/server/api/endpoints/admin/update-meta.ts +++ b/packages/backend/src/server/api/endpoints/admin/update-meta.ts @@ -126,6 +126,9 @@ export const paramDef = { deeplIsPro: { type: "boolean" }, libreTranslateApiUrl: { type: "string", nullable: true }, libreTranslateApiKey: { type: "string", nullable: true }, + enableGithubIntegration: { type: "boolean" }, + githubClientId: { type: "string", nullable: true }, + githubClientSecret: { type: "string", nullable: true }, enableEmail: { type: "boolean" }, email: { type: "string", nullable: true }, smtpSecure: { type: "boolean" }, diff --git a/packages/backend/src/server/api/endpoints/meta.ts b/packages/backend/src/server/api/endpoints/meta.ts index efce27f..c05b00f 100644 --- a/packages/backend/src/server/api/endpoints/meta.ts +++ b/packages/backend/src/server/api/endpoints/meta.ts @@ -287,6 +287,11 @@ export const meta = { optional: false, nullable: false, }, + enableGithubIntegration: { + type: "boolean", + optional: false, + nullable: false, + }, enableServiceWorker: { type: "boolean", optional: false, diff --git a/packages/backend/src/server/api/service/github.ts b/packages/backend/src/server/api/service/github.ts new file mode 100644 index 0000000..d829997 --- /dev/null +++ b/packages/backend/src/server/api/service/github.ts @@ -0,0 +1,295 @@ +import type Koa from "koa"; +import Router from "@koa/router"; +import { OAuth2 } from "oauth"; +import { v4 as uuid } from "uuid"; +import { IsNull } from "typeorm"; +import { getJson } from "@/misc/fetch.js"; +import config from "@/config/index.js"; +import { publishMainStream } from "@/services/stream.js"; +import { fetchMeta } from "@/misc/fetch-meta.js"; +import { Users, UserProfiles } from "@/models/index.js"; +import type { ILocalUser } from "@/models/entities/user.js"; +import { redisClient } from "../../../db/redis.js"; +import signin from "../common/signin.js"; + +function getUserToken(ctx: Koa.BaseContext): string | null { + return ((ctx.headers["cookie"] || "").match(/igi=(\w+)/) || [null, null])[1]; +} + +function compareOrigin(ctx: Koa.BaseContext): boolean { + function normalizeUrl(url?: string): string { + return url ? (url.endsWith("/") ? url.slice(0, url.length - 1) : url) : ""; + } + + const referer = ctx.headers["referer"]; + + return normalizeUrl(referer) === normalizeUrl(config.url); +} + +// Init router +const router = new Router(); + +router.get("/disconnect/github", async (ctx) => { + if (!compareOrigin(ctx)) { + ctx.throw(400, "invalid origin"); + return; + } + + const userToken = getUserToken(ctx); + if (!userToken) { + ctx.throw(400, "signin required"); + return; + } + + const user = await Users.findOneByOrFail({ + host: IsNull(), + token: userToken, + }); + + const profile = await UserProfiles.findOneByOrFail({ userId: user.id }); + + profile.integrations.github = undefined; + + await UserProfiles.update(user.id, { + integrations: profile.integrations, + }); + + ctx.body = "El enlace de GitHub ha sido cancelado. :v:"; + + // Publish i updated event + publishMainStream( + user.id, + "meUpdated", + await Users.pack(user, user, { + detail: true, + includeSecrets: true, + }), + ); +}); + +async function getOath2() { + const meta = await fetchMeta(true); + + if ( + meta.enableGithubIntegration && + meta.githubClientId && + meta.githubClientSecret + ) { + return new OAuth2( + meta.githubClientId, + meta.githubClientSecret, + "https://github.com/", + "login/oauth/authorize", + "login/oauth/access_token", + ); + } else { + return null; + } +} + +router.get("/connect/github", async (ctx) => { + if (!compareOrigin(ctx)) { + ctx.throw(400, "invalid origin"); + return; + } + + const userToken = getUserToken(ctx); + if (!userToken) { + ctx.throw(400, "signin required"); + return; + } + + const params = { + redirect_uri: `${config.url}/api/gh/cb`, + scope: ["read:user"], + state: uuid(), + }; + + redisClient.set(userToken, JSON.stringify(params)); + + const oauth2 = await getOath2(); + ctx.redirect(oauth2!.getAuthorizeUrl(params)); +}); + +router.get("/signin/github", async (ctx) => { + const sessid = uuid(); + + const params = { + redirect_uri: `${config.url}/api/gh/cb`, + scope: ["read:user"], + state: uuid(), + }; + + ctx.cookies.set("signin_with_github_sid", sessid, { + path: "/", + secure: config.url.startsWith("https"), + httpOnly: true, + }); + + redisClient.set(sessid, JSON.stringify(params)); + + const oauth2 = await getOath2(); + ctx.redirect(oauth2!.getAuthorizeUrl(params)); +}); + +router.get("/gh/cb", async (ctx) => { + const userToken = getUserToken(ctx); + + const oauth2 = await getOath2(); + + if (!userToken) { + const sessid = ctx.cookies.get("signin_with_github_sid"); + + if (!sessid) { + ctx.throw(400, "invalid session"); + return; + } + + const code = ctx.query.code; + + if (!code || typeof code !== "string") { + ctx.throw(400, "invalid session"); + return; + } + + const { redirect_uri, state } = await new Promise((res, rej) => { + redisClient.get(sessid, async (_, state) => { + res(JSON.parse(state)); + }); + }); + + if (ctx.query.state !== state) { + ctx.throw(400, "invalid session"); + return; + } + + const { accessToken } = await new Promise((res, rej) => + oauth2!.getOAuthAccessToken( + code, + { + redirect_uri, + }, + (err, accessToken, refresh, result) => { + if (err) { + rej(err); + } else if (result.error) { + rej(result.error); + } else { + res({ accessToken }); + } + }, + ), + ); + + const { login, id } = (await getJson( + "https://api.github.com/user", + "application/vnd.github.v3+json", + 10 * 1000, + { + Authorization: `bearer ${accessToken}`, + }, + )) as Record; + if (typeof login !== "string" || typeof id !== "string") { + ctx.throw(400, "invalid session"); + return; + } + + const link = await UserProfiles.createQueryBuilder() + .where("\"integrations\"->'github'->>'id' = :id", { id: id }) + .andWhere('"userHost" IS NULL') + .getOne(); + + if (link == null) { + ctx.throw( + 404, + `@${login}No había ninguna cuenta de Fedired vinculada con...`, + ); + return; + } + + signin( + ctx, + (await Users.findOneBy({ id: link.userId })) as ILocalUser, + true, + ); + } else { + const code = ctx.query.code; + + if (!code || typeof code !== "string") { + ctx.throw(400, "invalid session"); + return; + } + + const { redirect_uri, state } = await new Promise((res, rej) => { + redisClient.get(userToken, async (_, state) => { + res(JSON.parse(state)); + }); + }); + + if (ctx.query.state !== state) { + ctx.throw(400, "invalid session"); + return; + } + + const { accessToken } = await new Promise((res, rej) => + oauth2!.getOAuthAccessToken( + code, + { redirect_uri }, + (err, accessToken, refresh, result) => { + if (err) { + rej(err); + } else if (result.error) { + rej(result.error); + } else { + res({ accessToken }); + } + }, + ), + ); + + const { login, id } = (await getJson( + "https://api.github.com/user", + "application/vnd.github.v3+json", + 10 * 1000, + { + Authorization: `bearer ${accessToken}`, + }, + )) as Record; + + if (typeof login !== "string" || typeof id !== "string") { + ctx.throw(400, "invalid session"); + return; + } + + const user = await Users.findOneByOrFail({ + host: IsNull(), + token: userToken, + }); + + const profile = await UserProfiles.findOneByOrFail({ userId: user.id }); + + await UserProfiles.update(user.id, { + integrations: { + ...profile.integrations, + github: { + accessToken: accessToken, + id: id, + login: login, + }, + }, + }); + + ctx.body = `GitHub: @${login} conectado a Fedired: @${user.username}! `; + // Publish i updated event + publishMainStream( + user.id, + "meUpdated", + await Users.pack(user, user, { + detail: true, + includeSecrets: true, + }), + ); + } +}); + +export default router; diff --git a/packages/client/src/components/MkSignin.vue b/packages/client/src/components/MkSignin.vue index 0f4ec6e..9ba5883 100644 --- a/packages/client/src/components/MkSignin.vue +++ b/packages/client/src/components/MkSignin.vue @@ -130,6 +130,18 @@ + diff --git a/packages/client/src/components/note/MkNoteHeader.vue b/packages/client/src/components/note/MkNoteHeader.vue index 7d4fbe9..56c1e4d 100644 --- a/packages/client/src/components/note/MkNoteHeader.vue +++ b/packages/client/src/components/note/MkNoteHeader.vue @@ -92,14 +92,13 @@ function openServerInfo() { justify-self: flex-end; border-radius: 100px; font-size: 0.8em; - text-shadow: 0 2px 2px var(--shadow); > .avatar { - inline-size: 3.7em; - block-size: 3.7em; - margin-inline-end: 1em; + width: 3.7em; + height: 3.7em; + margin-right: 1em; } > .user-info { - inline-size: 0; + width: 0; flex-grow: 1; line-height: 1.5; display: flex; @@ -107,13 +106,13 @@ function openServerInfo() { > div { &:first-child { flex-grow: 1; - inline-size: 0; + width: 0; overflow: hidden; text-overflow: ellipsis; gap: 0.1em 0; } &:last-child { - max-inline-size: 50%; + max-width: 50%; gap: 0.3em 0.5em; } .article > .main & { @@ -124,17 +123,14 @@ function openServerInfo() { align-items: flex-end; } > * { - max-inline-size: 100%; + max-width: 100%; } } } .name { // flex: 1 1 0px; display: inline; - margin-block-start: 0; - margin-inline-end: 0.5em; - margin-block-end: 0; - margin-inline-start: 0; + margin: 0 0.5em 0 0; padding: 0; overflow: hidden; font-weight: bold; @@ -144,12 +140,8 @@ function openServerInfo() { .mkusername > .is-bot { flex-shrink: 0; align-self: center; - margin-block-start: 0; - margin-inline-end: 0.5em; - margin-block-end: 0; - margin-inline-start: 0; - padding-block: 1px; - padding-inline: 6px; + margin: 0 0.5em 0 0; + padding: 1px 6px; font-size: 80%; border: solid 0.5px var(--divider); border-radius: 3px; @@ -162,10 +154,7 @@ function openServerInfo() { .username { display: inline; - margin-block-start: 0; - margin-inline-end: 0.5em; - margin-block-end: 0; - margin-inline-start: 0; + margin: 0 0.5em 0 0; overflow: hidden; text-overflow: ellipsis; align-self: flex-start; @@ -175,10 +164,10 @@ function openServerInfo() { .info { display: inline-flex; flex-shrink: 0; - margin-inline-start: 0.5em; + margin-left: 0.5em; font-size: 0.9em; .created-at { - max-inline-size: 100%; + max-width: 100%; overflow: hidden; text-overflow: ellipsis; } @@ -186,7 +175,7 @@ function openServerInfo() { .ticker { display: inline-flex; - margin-inline-start: 0.5em; + margin-left: 0.5em; vertical-align: middle; > .name { display: none; diff --git a/packages/client/src/pages/admin/integrations.github.vue b/packages/client/src/pages/admin/integrations.github.vue new file mode 100644 index 0000000..67c57a7 --- /dev/null +++ b/packages/client/src/pages/admin/integrations.github.vue @@ -0,0 +1,69 @@ + + + diff --git a/packages/client/src/pages/settings/integration.vue b/packages/client/src/pages/settings/integration.vue new file mode 100644 index 0000000..2b9b541 --- /dev/null +++ b/packages/client/src/pages/settings/integration.vue @@ -0,0 +1,67 @@ + + + \ No newline at end of file diff --git a/packages/fedired-js/src/entities.ts b/packages/fedired-js/src/entities.ts index 83ebc25..71a4a47 100644 --- a/packages/fedired-js/src/entities.ts +++ b/packages/fedired-js/src/entities.ts @@ -370,6 +370,7 @@ export type LiteInstanceMetadata = { swPublickey: string | null; maxNoteTextLength: number; enableEmail: boolean; + enableGithubIntegration: boolean; enableServiceWorker: boolean; markLocalFilesNsfwByDefault: boolean; emojis: CustomEmoji[]; From c05d5de28d2945033ee2a42d8cf35a7caa8231f5 Mon Sep 17 00:00:00 2001 From: Javier Caceres <81489497+srnovus@users.noreply.github.com> Date: Wed, 30 Oct 2024 19:25:07 -0600 Subject: [PATCH 4/5] Nvus (#25) * add ci * add github yml * update ci * update ci * update error ci * update * resolver issue #6 * ci * udpate ci * update ci * a * add github integration and update mknote * update * update code --- .../server/api/endpoints/admin/update-meta.ts | 12 +++++ .../client/src/pages/admin/integrations.vue | 52 +++++++++++++++++++ 2 files changed, 64 insertions(+) create mode 100644 packages/client/src/pages/admin/integrations.vue diff --git a/packages/backend/src/server/api/endpoints/admin/update-meta.ts b/packages/backend/src/server/api/endpoints/admin/update-meta.ts index 01f1fe2..f31ead3 100644 --- a/packages/backend/src/server/api/endpoints/admin/update-meta.ts +++ b/packages/backend/src/server/api/endpoints/admin/update-meta.ts @@ -400,6 +400,18 @@ export default define(meta, paramDef, async (ps, me) => { set.summalyProxy = ps.summalyProxy; } + + if (ps.enableGithubIntegration !== undefined) { + set.enableGithubIntegration = ps.enableGithubIntegration; + } + + if (ps.githubClientId !== undefined) { + set.githubClientId = ps.githubClientId; + } + + if (ps.githubClientSecret !== undefined) { + set.githubClientSecret = ps.githubClientSecret; + } if (ps.enableEmail !== undefined) { set.enableEmail = ps.enableEmail; } diff --git a/packages/client/src/pages/admin/integrations.vue b/packages/client/src/pages/admin/integrations.vue new file mode 100644 index 0000000..acab68e --- /dev/null +++ b/packages/client/src/pages/admin/integrations.vue @@ -0,0 +1,52 @@ + + + \ No newline at end of file From 5b04fe92fcbcaa854e8db29e2b85ea4be4bad9fb Mon Sep 17 00:00:00 2001 From: Javier Caceres <81489497+srnovus@users.noreply.github.com> Date: Wed, 30 Oct 2024 19:40:12 -0600 Subject: [PATCH 5/5] Nvus (#26) * add ci * add github yml * update ci * update ci * update error ci * update * resolver issue #6 * ci * udpate ci * update ci * a * add github integration and update mknote * update * update code * update --- packages/client/src/pages/admin/index.vue | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/packages/client/src/pages/admin/index.vue b/packages/client/src/pages/admin/index.vue index 3f3bf2f..17a699b 100644 --- a/packages/client/src/pages/admin/index.vue +++ b/packages/client/src/pages/admin/index.vue @@ -258,6 +258,12 @@ const menuDef = computed(() => [ to: "/admin/relays", active: currentPage.value?.route.name === "relays", }, + { + icon: "ph-plug ph-bold ph-lg", + text: i18n.ts.integration, + to: "/admin/integrations", + active: currentPage?.route.name === "integrations", + }, { icon: `${icon("ph-prohibit")}`, text: i18n.ts.instanceBlocking,