2023-07-28 07:02:58 +02:00
|
|
|
/*
|
|
|
|
* SPDX-FileCopyrightText: syuilo and other misskey contributors
|
|
|
|
* SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
*/
|
|
|
|
|
2023-07-27 11:51:58 +02:00
|
|
|
import { Inject, Injectable } from '@nestjs/common';
|
2023-09-24 19:47:12 +02:00
|
|
|
import megalodon, { MegalodonInterface } from 'megalodon';
|
2023-09-24 23:18:39 +02:00
|
|
|
import querystring from 'querystring';
|
2023-09-24 19:47:12 +02:00
|
|
|
import { v4 as uuid } from 'uuid';
|
|
|
|
/* import { kinds } from '@/misc/api-permissions.js';
|
2023-07-27 11:51:58 +02:00
|
|
|
import type { Config } from '@/config.js';
|
2023-09-24 19:47:12 +02:00
|
|
|
import { DI } from '@/di-symbols.js'; */
|
2023-07-27 11:51:58 +02:00
|
|
|
import { bindThis } from '@/decorators.js';
|
|
|
|
import type { FastifyInstance } from 'fastify';
|
|
|
|
|
2023-09-24 23:18:39 +02:00
|
|
|
function getClient(BASE_URL: string, authorization: string | undefined): MegalodonInterface {
|
|
|
|
const accessTokenArr = authorization?.split(' ') ?? [null];
|
|
|
|
const accessToken = accessTokenArr[accessTokenArr.length - 1];
|
|
|
|
const generator = (megalodon as any).default;
|
|
|
|
const client = generator('misskey', BASE_URL, accessToken) as MegalodonInterface;
|
|
|
|
return client;
|
|
|
|
}
|
|
|
|
|
2023-07-27 11:51:58 +02:00
|
|
|
@Injectable()
|
|
|
|
export class OAuth2ProviderService {
|
|
|
|
constructor(
|
2023-09-24 19:47:12 +02:00
|
|
|
/* @Inject(DI.config)
|
|
|
|
private config: Config, */
|
|
|
|
) { }
|
2023-07-27 11:51:58 +02:00
|
|
|
|
|
|
|
@bindThis
|
|
|
|
public async createServer(fastify: FastifyInstance): Promise<void> {
|
|
|
|
// https://datatracker.ietf.org/doc/html/rfc8414.html
|
|
|
|
// https://indieauth.spec.indieweb.org/#indieauth-server-metadata
|
2023-09-24 19:47:12 +02:00
|
|
|
/* fastify.get('/.well-known/oauth-authorization-server', async (_request, reply) => {
|
2023-07-27 11:51:58 +02:00
|
|
|
reply.send({
|
|
|
|
issuer: this.config.url,
|
|
|
|
authorization_endpoint: new URL('/oauth/authorize', this.config.url),
|
|
|
|
token_endpoint: new URL('/oauth/token', this.config.url),
|
|
|
|
scopes_supported: kinds,
|
|
|
|
response_types_supported: ['code'],
|
|
|
|
grant_types_supported: ['authorization_code'],
|
|
|
|
service_documentation: 'https://misskey-hub.net',
|
|
|
|
code_challenge_methods_supported: ['S256'],
|
|
|
|
authorization_response_iss_parameter_supported: true,
|
|
|
|
});
|
2023-09-24 19:47:12 +02:00
|
|
|
}); */
|
2023-07-27 11:51:58 +02:00
|
|
|
|
2023-09-24 23:01:46 +02:00
|
|
|
fastify.addHook('onRequest', (request, reply, done) => {
|
|
|
|
reply.header('Access-Control-Allow-Origin', '*');
|
|
|
|
done();
|
|
|
|
});
|
|
|
|
|
2023-09-24 23:18:39 +02:00
|
|
|
fastify.addContentTypeParser('application/x-www-form-urlencoded', function (request, payload, done) {
|
|
|
|
let body = '';
|
|
|
|
payload.on('data', function (data) {
|
|
|
|
body += data;
|
|
|
|
});
|
|
|
|
payload.on('end', function () {
|
|
|
|
try {
|
|
|
|
const parsed = querystring.parse(body);
|
|
|
|
done(null, parsed);
|
|
|
|
} catch (e: any) {
|
|
|
|
done(e);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
payload.on('error', done);
|
2023-09-24 23:01:46 +02:00
|
|
|
});
|
|
|
|
|
2023-07-27 11:51:58 +02:00
|
|
|
fastify.get('/oauth/authorize', async (request, reply) => {
|
2023-09-24 19:47:12 +02:00
|
|
|
const query: any = request.query;
|
|
|
|
let param = "mastodon=true";
|
|
|
|
if (query.state) param += `&state=${query.state}`;
|
|
|
|
if (query.redirect_uri) param += `&redirect_uri=${query.redirect_uri}`;
|
|
|
|
const client = query.client_id ? query.client_id : "";
|
|
|
|
reply.redirect(
|
|
|
|
`${atob(client)}?${param}`,
|
|
|
|
);
|
2023-07-27 11:51:58 +02:00
|
|
|
});
|
|
|
|
|
2023-09-24 19:47:12 +02:00
|
|
|
fastify.post('/oauth/token', async (request, reply) => {
|
|
|
|
const body: any = request.body || request.query;
|
|
|
|
if (body.grant_type === "client_credentials") {
|
|
|
|
const ret = {
|
|
|
|
access_token: uuid(),
|
|
|
|
token_type: "Bearer",
|
|
|
|
scope: "read",
|
|
|
|
created_at: Math.floor(new Date().getTime() / 1000),
|
|
|
|
};
|
|
|
|
reply.send(ret);
|
|
|
|
}
|
|
|
|
let client_id: any = body.client_id;
|
|
|
|
const BASE_URL = `${request.protocol}://${request.hostname}`;
|
2023-09-24 23:18:39 +02:00
|
|
|
const client = getClient(BASE_URL, '');
|
2023-09-24 19:47:12 +02:00
|
|
|
let token = null;
|
|
|
|
if (body.code) {
|
|
|
|
//m = body.code.match(/^([a-zA-Z0-9]{8})([a-zA-Z0-9]{4})([a-zA-Z0-9]{4})([a-zA-Z0-9]{4})([a-zA-Z0-9]{12})/);
|
|
|
|
//if (!m.length) {
|
|
|
|
// ctx.body = { error: "Invalid code" };
|
|
|
|
// return;
|
|
|
|
//}
|
|
|
|
//token = `${m[1]}-${m[2]}-${m[3]}-${m[4]}-${m[5]}`
|
|
|
|
//console.log(body.code, token);
|
|
|
|
token = body.code;
|
|
|
|
}
|
|
|
|
if (client_id instanceof Array) {
|
|
|
|
client_id = client_id.toString();
|
|
|
|
} else if (!client_id) {
|
|
|
|
client_id = null;
|
|
|
|
}
|
|
|
|
try {
|
|
|
|
const atData = await client.fetchAccessToken(
|
|
|
|
client_id,
|
|
|
|
body.client_secret,
|
|
|
|
token ? token : "",
|
|
|
|
);
|
|
|
|
const ret = {
|
|
|
|
access_token: atData.accessToken,
|
|
|
|
token_type: "Bearer",
|
|
|
|
scope: body.scope || "read write follow push",
|
|
|
|
created_at: Math.floor(new Date().getTime() / 1000),
|
|
|
|
};
|
|
|
|
reply.send(ret);
|
|
|
|
} catch (err: any) {
|
2023-09-24 23:18:39 +02:00
|
|
|
console.error(err);
|
|
|
|
reply.code(401).send(err.response.data);
|
2023-09-24 19:47:12 +02:00
|
|
|
}
|
2023-07-27 11:51:58 +02:00
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|