mirror of
https://codeberg.org/yeentown/barkey.git
synced 2025-08-21 18:43:37 +00:00
implement '/v1/apps/verify_credentials'
This commit is contained in:
parent
5ec9be0b8c
commit
b2ea03383c
14 changed files with 183 additions and 9 deletions
|
@ -84,6 +84,8 @@ export class AuthenticateService implements OnApplicationShutdown {
|
||||||
return [user, {
|
return [user, {
|
||||||
id: accessToken.id,
|
id: accessToken.id,
|
||||||
permission: app.permission,
|
permission: app.permission,
|
||||||
|
appId: app.id,
|
||||||
|
app,
|
||||||
} as MiAccessToken];
|
} as MiAccessToken];
|
||||||
} else {
|
} else {
|
||||||
return [user, accessToken];
|
return [user, accessToken];
|
||||||
|
|
|
@ -128,6 +128,7 @@ export * as 'antennas/update' from './endpoints/antennas/update.js';
|
||||||
export * as 'ap/get' from './endpoints/ap/get.js';
|
export * as 'ap/get' from './endpoints/ap/get.js';
|
||||||
export * as 'ap/show' from './endpoints/ap/show.js';
|
export * as 'ap/show' from './endpoints/ap/show.js';
|
||||||
export * as 'app/create' from './endpoints/app/create.js';
|
export * as 'app/create' from './endpoints/app/create.js';
|
||||||
|
export * as 'app/current' from './endpoints/app/current.js';
|
||||||
export * as 'app/show' from './endpoints/app/show.js';
|
export * as 'app/show' from './endpoints/app/show.js';
|
||||||
export * as 'auth/accept' from './endpoints/auth/accept.js';
|
export * as 'auth/accept' from './endpoints/auth/accept.js';
|
||||||
export * as 'auth/session/generate' from './endpoints/auth/session/generate.js';
|
export * as 'auth/session/generate' from './endpoints/auth/session/generate.js';
|
||||||
|
|
73
packages/backend/src/server/api/endpoints/app/current.ts
Normal file
73
packages/backend/src/server/api/endpoints/app/current.ts
Normal file
|
@ -0,0 +1,73 @@
|
||||||
|
/*
|
||||||
|
* SPDX-FileCopyrightText: hazelnoot and other Sharkey contributors
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { Inject, Injectable } from '@nestjs/common';
|
||||||
|
import { Endpoint } from '@/server/api/endpoint-base.js';
|
||||||
|
import type { AppsRepository } from '@/models/_.js';
|
||||||
|
import { AppEntityService } from '@/core/entities/AppEntityService.js';
|
||||||
|
import { DI } from '@/di-symbols.js';
|
||||||
|
import { ApiError } from '../../error.js';
|
||||||
|
|
||||||
|
export const meta = {
|
||||||
|
tags: ['app'],
|
||||||
|
|
||||||
|
errors: {
|
||||||
|
credentialRequired: {
|
||||||
|
message: 'Credential required.',
|
||||||
|
code: 'CREDENTIAL_REQUIRED',
|
||||||
|
id: '1384574d-a912-4b81-8601-c7b1c4085df1',
|
||||||
|
httpStatusCode: 401,
|
||||||
|
},
|
||||||
|
noAppLogin: {
|
||||||
|
message: 'Not logged in with an app.',
|
||||||
|
code: 'NO_APP_LOGIN',
|
||||||
|
id: '339a4ad2-48c3-47fc-bd9d-2408f05120f8',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
|
res: {
|
||||||
|
type: 'object',
|
||||||
|
optional: false, nullable: false,
|
||||||
|
ref: 'App',
|
||||||
|
},
|
||||||
|
|
||||||
|
// 10 calls per 5 seconds
|
||||||
|
limit: {
|
||||||
|
duration: 1000 * 5,
|
||||||
|
max: 10,
|
||||||
|
},
|
||||||
|
} as const;
|
||||||
|
|
||||||
|
export const paramDef = {
|
||||||
|
type: 'object',
|
||||||
|
properties: {},
|
||||||
|
required: [],
|
||||||
|
} as const;
|
||||||
|
|
||||||
|
@Injectable()
|
||||||
|
export default class extends Endpoint<typeof meta, typeof paramDef> { // eslint-disable-line import/no-default-export
|
||||||
|
constructor(
|
||||||
|
@Inject(DI.appsRepository)
|
||||||
|
private appsRepository: AppsRepository,
|
||||||
|
|
||||||
|
private appEntityService: AppEntityService,
|
||||||
|
) {
|
||||||
|
super(meta, paramDef, async (_, user, token) => {
|
||||||
|
if (!user) {
|
||||||
|
throw new ApiError(meta.errors.credentialRequired);
|
||||||
|
}
|
||||||
|
if (!token || !token.appId) {
|
||||||
|
throw new ApiError(meta.errors.noAppLogin);
|
||||||
|
}
|
||||||
|
|
||||||
|
const app = token.app ?? await this.appsRepository.findOneByOrFail({ id: token.appId });
|
||||||
|
|
||||||
|
return await this.appEntityService.pack(app, user, {
|
||||||
|
detail: true,
|
||||||
|
includeSecret: false,
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
|
@ -4,7 +4,7 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import { Inject, Injectable } from '@nestjs/common';
|
import { Inject, Injectable } from '@nestjs/common';
|
||||||
import { Entity, MastodonEntity } from 'megalodon';
|
import { Entity, MastodonEntity, MisskeyEntity } from 'megalodon';
|
||||||
import mfm from '@transfem-org/sfm-js';
|
import mfm from '@transfem-org/sfm-js';
|
||||||
import { MastodonNotificationType } from 'megalodon/lib/src/mastodon/notification.js';
|
import { MastodonNotificationType } from 'megalodon/lib/src/mastodon/notification.js';
|
||||||
import { NotificationType } from 'megalodon/lib/src/notification.js';
|
import { NotificationType } from 'megalodon/lib/src/notification.js';
|
||||||
|
@ -369,6 +369,15 @@ export class MastodonConverters {
|
||||||
type: convertNotificationType(notification.type as NotificationType),
|
type: convertNotificationType(notification.type as NotificationType),
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public convertApplication(app: MisskeyEntity.App): MastodonEntity.Application {
|
||||||
|
return {
|
||||||
|
name: app.name,
|
||||||
|
scopes: app.permission,
|
||||||
|
redirect_uri: app.callbackUrl,
|
||||||
|
redirect_uris: [app.callbackUrl],
|
||||||
|
};
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function simpleConvert<T>(data: T): T {
|
function simpleConvert<T>(data: T): T {
|
||||||
|
@ -459,4 +468,3 @@ export function convertRelationship(relationship: Partial<Entity.Relationship> &
|
||||||
note: relationship.note ?? '',
|
note: relationship.note ?? '',
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -5,6 +5,7 @@
|
||||||
|
|
||||||
import { Injectable } from '@nestjs/common';
|
import { Injectable } from '@nestjs/common';
|
||||||
import { MastodonClientService } from '@/server/api/mastodon/MastodonClientService.js';
|
import { MastodonClientService } from '@/server/api/mastodon/MastodonClientService.js';
|
||||||
|
import { MastodonConverters } from '@/server/api/mastodon/MastodonConverters.js';
|
||||||
import type { FastifyInstance } from 'fastify';
|
import type { FastifyInstance } from 'fastify';
|
||||||
|
|
||||||
const readScope = [
|
const readScope = [
|
||||||
|
@ -59,6 +60,7 @@ type AuthMastodonRoute = { Body?: AuthPayload, Querystring: AuthPayload };
|
||||||
export class ApiAppsMastodon {
|
export class ApiAppsMastodon {
|
||||||
constructor(
|
constructor(
|
||||||
private readonly clientService: MastodonClientService,
|
private readonly clientService: MastodonClientService,
|
||||||
|
private readonly mastoConverters: MastodonConverters,
|
||||||
) {}
|
) {}
|
||||||
|
|
||||||
public register(fastify: FastifyInstance): void {
|
public register(fastify: FastifyInstance): void {
|
||||||
|
@ -108,6 +110,13 @@ export class ApiAppsMastodon {
|
||||||
|
|
||||||
return reply.send(response);
|
return reply.send(response);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
fastify.get('/v1/apps/verify_credentials', async (_request, reply) => {
|
||||||
|
const client = this.clientService.getClient(_request);
|
||||||
|
const data = await client.verifyAppCredentials();
|
||||||
|
const response = this.mastoConverters.convertApplication(data.data);
|
||||||
|
return reply.send(response);
|
||||||
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -9,6 +9,7 @@ import * as NotificationType from './notification'
|
||||||
import FilterContext from './filter_context'
|
import FilterContext from './filter_context'
|
||||||
import Converter from './converter'
|
import Converter from './converter'
|
||||||
import MastodonEntity from './mastodon/entity';
|
import MastodonEntity from './mastodon/entity';
|
||||||
|
import MisskeyEntity from './misskey/entity';
|
||||||
|
|
||||||
export {
|
export {
|
||||||
Response,
|
Response,
|
||||||
|
@ -23,4 +24,5 @@ export {
|
||||||
Entity,
|
Entity,
|
||||||
Converter,
|
Converter,
|
||||||
MastodonEntity,
|
MastodonEntity,
|
||||||
|
MisskeyEntity,
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,5 +3,8 @@ namespace MastodonEntity {
|
||||||
name: string
|
name: string
|
||||||
website?: string | null
|
website?: string | null
|
||||||
vapid_key?: string | null
|
vapid_key?: string | null
|
||||||
|
scopes: string[]
|
||||||
|
redirect_uris: string[]
|
||||||
|
redirect_uri?: string
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -102,7 +102,7 @@ export default class Misskey implements MegalodonInterface {
|
||||||
website: null,
|
website: null,
|
||||||
redirect_uri: res.data.callbackUrl,
|
redirect_uri: res.data.callbackUrl,
|
||||||
client_id: '',
|
client_id: '',
|
||||||
client_secret: res.data.secret
|
client_secret: res.data.secret!
|
||||||
}
|
}
|
||||||
return OAuth.AppData.from(appData)
|
return OAuth.AppData.from(appData)
|
||||||
})
|
})
|
||||||
|
@ -122,11 +122,8 @@ export default class Misskey implements MegalodonInterface {
|
||||||
// ======================================
|
// ======================================
|
||||||
// apps
|
// apps
|
||||||
// ======================================
|
// ======================================
|
||||||
public async verifyAppCredentials(): Promise<Response<Entity.Application>> {
|
public async verifyAppCredentials(): Promise<Response<MisskeyAPI.Entity.App>> {
|
||||||
return new Promise((_, reject) => {
|
return await this.client.post<MisskeyAPI.Entity.App>('/api/app/current');
|
||||||
const err = new NoImplementedError('misskey does not support')
|
|
||||||
reject(err)
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// ======================================
|
// ======================================
|
||||||
|
|
|
@ -4,6 +4,6 @@ namespace MisskeyEntity {
|
||||||
name: string
|
name: string
|
||||||
callbackUrl: string
|
callbackUrl: string
|
||||||
permission: Array<string>
|
permission: Array<string>
|
||||||
secret: string
|
secret?: string
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -552,6 +552,9 @@ type AppCreateRequest = operations['app___create']['requestBody']['content']['ap
|
||||||
// @public (undocumented)
|
// @public (undocumented)
|
||||||
type AppCreateResponse = operations['app___create']['responses']['200']['content']['application/json'];
|
type AppCreateResponse = operations['app___create']['responses']['200']['content']['application/json'];
|
||||||
|
|
||||||
|
// @public (undocumented)
|
||||||
|
type AppCurrentResponse = operations['app___current']['responses']['200']['content']['application/json'];
|
||||||
|
|
||||||
// @public (undocumented)
|
// @public (undocumented)
|
||||||
type AppShowRequest = operations['app___show']['requestBody']['content']['application/json'];
|
type AppShowRequest = operations['app___show']['requestBody']['content']['application/json'];
|
||||||
|
|
||||||
|
@ -1643,6 +1646,7 @@ declare namespace entities {
|
||||||
ApShowResponse,
|
ApShowResponse,
|
||||||
AppCreateRequest,
|
AppCreateRequest,
|
||||||
AppCreateResponse,
|
AppCreateResponse,
|
||||||
|
AppCurrentResponse,
|
||||||
AppShowRequest,
|
AppShowRequest,
|
||||||
AppShowResponse,
|
AppShowResponse,
|
||||||
AuthAcceptRequest,
|
AuthAcceptRequest,
|
||||||
|
|
|
@ -1313,6 +1313,17 @@ declare module '../api.js' {
|
||||||
credential?: string | null,
|
credential?: string | null,
|
||||||
): Promise<SwitchCaseResponseType<E, P>>;
|
): Promise<SwitchCaseResponseType<E, P>>;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* No description provided.
|
||||||
|
*
|
||||||
|
* **Credential required**: *No*
|
||||||
|
*/
|
||||||
|
request<E extends 'app/current', P extends Endpoints[E]['req']>(
|
||||||
|
endpoint: E,
|
||||||
|
params: P,
|
||||||
|
credential?: string | null,
|
||||||
|
): Promise<SwitchCaseResponseType<E, P>>;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* No description provided.
|
* No description provided.
|
||||||
*
|
*
|
||||||
|
|
|
@ -159,6 +159,7 @@ import type {
|
||||||
ApShowResponse,
|
ApShowResponse,
|
||||||
AppCreateRequest,
|
AppCreateRequest,
|
||||||
AppCreateResponse,
|
AppCreateResponse,
|
||||||
|
AppCurrentResponse,
|
||||||
AppShowRequest,
|
AppShowRequest,
|
||||||
AppShowResponse,
|
AppShowResponse,
|
||||||
AuthAcceptRequest,
|
AuthAcceptRequest,
|
||||||
|
@ -778,6 +779,7 @@ export type Endpoints = {
|
||||||
'ap/get': { req: ApGetRequest; res: ApGetResponse };
|
'ap/get': { req: ApGetRequest; res: ApGetResponse };
|
||||||
'ap/show': { req: ApShowRequest; res: ApShowResponse };
|
'ap/show': { req: ApShowRequest; res: ApShowResponse };
|
||||||
'app/create': { req: AppCreateRequest; res: AppCreateResponse };
|
'app/create': { req: AppCreateRequest; res: AppCreateResponse };
|
||||||
|
'app/current': { req: EmptyRequest; res: AppCurrentResponse };
|
||||||
'app/show': { req: AppShowRequest; res: AppShowResponse };
|
'app/show': { req: AppShowRequest; res: AppShowResponse };
|
||||||
'auth/accept': { req: AuthAcceptRequest; res: EmptyResponse };
|
'auth/accept': { req: AuthAcceptRequest; res: EmptyResponse };
|
||||||
'auth/session/generate': { req: AuthSessionGenerateRequest; res: AuthSessionGenerateResponse };
|
'auth/session/generate': { req: AuthSessionGenerateRequest; res: AuthSessionGenerateResponse };
|
||||||
|
|
|
@ -162,6 +162,7 @@ export type ApShowRequest = operations['ap___show']['requestBody']['content']['a
|
||||||
export type ApShowResponse = operations['ap___show']['responses']['200']['content']['application/json'];
|
export type ApShowResponse = operations['ap___show']['responses']['200']['content']['application/json'];
|
||||||
export type AppCreateRequest = operations['app___create']['requestBody']['content']['application/json'];
|
export type AppCreateRequest = operations['app___create']['requestBody']['content']['application/json'];
|
||||||
export type AppCreateResponse = operations['app___create']['responses']['200']['content']['application/json'];
|
export type AppCreateResponse = operations['app___create']['responses']['200']['content']['application/json'];
|
||||||
|
export type AppCurrentResponse = operations['app___current']['responses']['200']['content']['application/json'];
|
||||||
export type AppShowRequest = operations['app___show']['requestBody']['content']['application/json'];
|
export type AppShowRequest = operations['app___show']['requestBody']['content']['application/json'];
|
||||||
export type AppShowResponse = operations['app___show']['responses']['200']['content']['application/json'];
|
export type AppShowResponse = operations['app___show']['responses']['200']['content']['application/json'];
|
||||||
export type AuthAcceptRequest = operations['auth___accept']['requestBody']['content']['application/json'];
|
export type AuthAcceptRequest = operations['auth___accept']['requestBody']['content']['application/json'];
|
||||||
|
|
|
@ -1086,6 +1086,15 @@ export type paths = {
|
||||||
*/
|
*/
|
||||||
post: operations['app___create'];
|
post: operations['app___create'];
|
||||||
};
|
};
|
||||||
|
'/app/current': {
|
||||||
|
/**
|
||||||
|
* app/current
|
||||||
|
* @description No description provided.
|
||||||
|
*
|
||||||
|
* **Credential required**: *No*
|
||||||
|
*/
|
||||||
|
post: operations['app___current'];
|
||||||
|
};
|
||||||
'/app/show': {
|
'/app/show': {
|
||||||
/**
|
/**
|
||||||
* app/show
|
* app/show
|
||||||
|
@ -13071,6 +13080,58 @@ export type operations = {
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
/**
|
||||||
|
* app/current
|
||||||
|
* @description No description provided.
|
||||||
|
*
|
||||||
|
* **Credential required**: *No*
|
||||||
|
*/
|
||||||
|
app___current: {
|
||||||
|
responses: {
|
||||||
|
/** @description OK (with results) */
|
||||||
|
200: {
|
||||||
|
content: {
|
||||||
|
'application/json': components['schemas']['App'];
|
||||||
|
};
|
||||||
|
};
|
||||||
|
/** @description Client error */
|
||||||
|
400: {
|
||||||
|
content: {
|
||||||
|
'application/json': components['schemas']['Error'];
|
||||||
|
};
|
||||||
|
};
|
||||||
|
/** @description Authentication error */
|
||||||
|
401: {
|
||||||
|
content: {
|
||||||
|
'application/json': components['schemas']['Error'];
|
||||||
|
};
|
||||||
|
};
|
||||||
|
/** @description Forbidden error */
|
||||||
|
403: {
|
||||||
|
content: {
|
||||||
|
'application/json': components['schemas']['Error'];
|
||||||
|
};
|
||||||
|
};
|
||||||
|
/** @description I'm Ai */
|
||||||
|
418: {
|
||||||
|
content: {
|
||||||
|
'application/json': components['schemas']['Error'];
|
||||||
|
};
|
||||||
|
};
|
||||||
|
/** @description Too many requests */
|
||||||
|
429: {
|
||||||
|
content: {
|
||||||
|
'application/json': components['schemas']['Error'];
|
||||||
|
};
|
||||||
|
};
|
||||||
|
/** @description Internal server error */
|
||||||
|
500: {
|
||||||
|
content: {
|
||||||
|
'application/json': components['schemas']['Error'];
|
||||||
|
};
|
||||||
|
};
|
||||||
|
};
|
||||||
|
};
|
||||||
/**
|
/**
|
||||||
* app/show
|
* app/show
|
||||||
* @description No description provided.
|
* @description No description provided.
|
||||||
|
|
Loading…
Add table
Reference in a new issue