Skip to content
Snippets Groups Projects
Verified Commit 387942fd authored by Konstantin Tsabolov's avatar Konstantin Tsabolov
Browse files

chore: remove Aries Mediator Service

parent 45595e8d
No related branches found
No related tags found
No related merge requests found
Showing
with 0 additions and 671 deletions
compose/aries-mediator-service/acapy/controller/public/favicon.ico

5.4 KiB

<!DOCTYPE html>
<html lang="en">
<head>
<title>A FeathersJS application</title>
<meta name="description" content="A FeathersJS server">
<meta name="viewport" content="width=device-width, initial-scale=1">
<style>
* {
margin: 0;
padding: 0;
box-sizing: border-box;
}
html, body {
font-family: 'Helvetica Neue', 'Helvetica', 'Arial', 'sans-serif';
font-weight: 400;
font-size: 16px;
color: #2c3e50;
}
.center-text {
text-align: center;
}
main {
margin-top: 100px;
padding: 20px;
}
img.logo {
display: block;
margin: 0 auto;
max-width: 100%;
margin-bottom: 30px;
}
h2 {
font-size: 2em;
font-weight: 100;
}
footer {
position: absolute;
bottom: 0;
left: 0;
right: 0;
padding: 20px;
}
footer p {
font-weight: 300;
font-size: 1.0em;
}
a {
color: #3cf;
text-decoration: none;
}
a:hover,
a:focus {
color: #3cf;
}
</style>
</head>
<body>
<main class="container">
<img class="logo" src="" alt="Feathers Logo">
<footer>
<p class="center-text">For more information on Feathers see <a href="https://docs.feathersjs.com" title="Feathers Documentation" target="blank">docs.feathersjs.com</a>.</p>
</footer>
</main>
</body>
</html>
// Application hooks that run for every service
// Don't remove this comment. It's needed to format import lines nicely.
export default {
before: {
all: [],
find: [],
get: [],
create: [],
update: [],
patch: [],
remove: [],
},
after: {
all: [],
find: [],
get: [],
create: [],
update: [],
patch: [],
remove: [],
},
error: {
all: [],
find: [],
get: [],
create: [],
update: [],
patch: [],
remove: [],
},
};
import path from 'path';
import favicon from 'serve-favicon';
import compress from 'compression';
import helmet from 'helmet';
import cors from 'cors';
import feathers from '@feathersjs/feathers';
import configuration from '@feathersjs/configuration';
import express from '@feathersjs/express';
import primus from '@feathersjs/primus';
import { Application } from './declarations';
import logger from './logger';
import middleware from './middleware';
import services from './services';
import appHooks from './app.hooks';
import channels from './channels';
import { HookContext as FeathersHookContext } from '@feathersjs/feathers';
// Don't remove this comment. It's needed to format import lines nicely.
const app: Application = express(feathers());
export type HookContext<T = any> = {
app: Application;
} & FeathersHookContext<T>;
// Load app configuration
app.configure(configuration());
// Enable security, CORS, compression, favicon and body parsing
app.use(
helmet({
contentSecurityPolicy: false,
}),
);
app.use(cors());
app.use(compress());
app.use(express.json());
app.use(express.urlencoded({ extended: true }));
app.use(favicon(path.join(app.get('public'), 'favicon.ico')));
// Host the public folder
app.use('/', express.static(app.get('public')));
// Set up Plugins and providers
app.configure(express.rest());
app.configure(primus({ transformer: 'websockets' }));
// Configure other middleware (see `middleware/index.ts`)
app.configure(middleware);
// Set up our services (see `services/index.ts`)
app.configure(services);
// Set up event channels (see channels.ts)
app.configure(channels);
// Configure a middleware for 404s and the error handler
app.use(express.notFound());
app.use(express.errorHandler({ logger } as any));
app.hooks(appHooks);
export default app;
import '@feathersjs/transport-commons';
import { HookContext } from '@feathersjs/feathers';
import { Application } from './declarations';
export default function (app: Application): void {
if (typeof app.channel !== 'function') {
// If no real-time functionality has been configured just return
return;
}
app.on('connection', (connection: any): void => {
// On a new real-time connection, add it to the anonymous channel
app.channel('anonymous').join(connection);
});
app.on('login', (authResult: any, { connection }: any): void => {
// connection can be undefined if there is no
// real-time connection, e.g. when logging in via REST
if (connection) {
// Obtain the logged in user from the connection
// const user = connection.user;
// The connection is no longer anonymous, remove it
app.channel('anonymous').leave(connection);
// Add it to the authenticated user channel
app.channel('authenticated').join(connection);
// Channels can be named anything and joined on any condition
// E.g. to send real-time events only to admins use
// if(user.isAdmin) { app.channel('admins').join(connection); }
// If the user has joined e.g. chat rooms
// if(Array.isArray(user.rooms)) user.rooms.forEach(room => app.channel(`rooms/${room.id}`).join(connection));
// Easily organize users by email and userid for things like messaging
// app.channel(`emails/${user.email}`).join(connection);
// app.channel(`userIds/${user.id}`).join(connection);
}
});
// eslint-disable-next-line @typescript-eslint/no-unused-vars
app.publish((data: any, hook: HookContext) => {
// Here you can add event publishers to channels set up in `channels.ts`
// To publish only for a specific event use `app.publish(eventname, () => {})`
// console.log('Publishing all events to all authenticated users. See `channels.ts` and https://docs.feathersjs.com/api/channels.html for more information.'); // eslint-disable-line
// e.g. to publish all service events to all authenticated users use
return app.channel('authenticated');
});
// Here you can also add service specific event publishers
// e.g. the publish the `users` service `created` event to the `admins` channel
// app.service('users').publish('created', () => app.channel('admins'));
// With the userid and email organization from above you can easily select involved users
// app.service('messages').publish(() => {
// return [
// app.channel(`userIds/${data.createdBy}`),
// app.channel(`emails/${data.recipientEmail}`)
// ];
// });
}
import { Application as ExpressFeathers } from '@feathersjs/express';
// A mapping of service names to types. Will be extended in service files.
export interface ServiceTypes {}
// The application instance type that will be used everywhere else
export type Application = ExpressFeathers<ServiceTypes>;
import app from './app';
import logger from './logger';
const port = app.get('port');
const server = app.listen(port);
process.on('unhandledRejection', (reason, p) =>
logger.error('Unhandled Rejection at: Promise ', p, reason),
);
process.on('SIGTERM', () => {
logger.warn('SIGTERM received, closing server.');
server.close(() => {
logger.warn('Server closed, exiting.');
process.exit(0);
});
});
server.on('listening', () =>
logger.info(
'Feathers application started on http://%s:%d',
app.get('host'),
port,
),
);
import { createLogger, format, transports } from 'winston';
// Configure the Winston logger. For the complete documentation see https://github.com/winstonjs/winston
const logger = createLogger({
// To see more detailed errors, change this to 'debug'
level: (process.env.LOG_LEVEL || 'info').toLowerCase(),
format: format.combine(format.splat(), format.simple()),
transports: [new transports.Console()],
});
export default logger;
import { Application } from '../declarations';
// Don't remove this comment. It's needed to format import lines nicely.
// eslint-disable-next-line @typescript-eslint/no-unused-vars, @typescript-eslint/no-empty-function
export default function (app: Application): void {}
export interface EndorserMetadataServiceRequest {
connection_id: string;
}
export enum ServiceType {
Connection = 'Connection',
}
export enum WebhookTopic {
Connections = 'connections',
}
export enum ConnectionState {
InvitationSent = 'invitation-sent',
Request = 'request',
Response = 'response',
Active = 'active',
Completed = 'completed',
}
export enum ConnectionServiceAction {
Accept_Connection_Request = 'Accept-Request',
Send_Connection_Ping = 'Send-Ping',
}
import { FeathersError } from '@feathersjs/errors';
export class UndefinedAppError extends Error {}
export class DuplicatedProfileError extends Error {}
export class AriesAgentError extends FeathersError {
constructor(
message: string | Error,
code: number | undefined,
data?: unknown,
) {
super(message, 'aries-agent-error', code || 500, 'AriesAgentError', data);
}
}
import { NotImplemented } from '@feathersjs/errors';
import { Params } from '@feathersjs/feathers';
import Axios, { AxiosError } from 'axios';
import { Application } from '../../declarations';
import logger from '../../logger';
import { ConnectionServiceAction, ServiceType } from '../../models/enums';
import { AriesAgentError } from '../../models/errors';
import { AcaPyUtils } from '../../utils/aca-py';
export interface AriesAgentData {
service: ServiceType;
action: ConnectionServiceAction;
data: any;
}
interface ServiceOptions {}
export class AriesAgent {
app: Application;
options: ServiceOptions;
acaPyUtils: AcaPyUtils;
constructor(options: ServiceOptions = {}, app: Application) {
this.options = options;
this.app = app;
this.acaPyUtils = AcaPyUtils.getInstance(app);
this.init();
}
private async init() {
await this.acaPyUtils.init();
logger.info('Aries Agent service initialized');
}
//eslint-disable-next-line @typescript-eslint/no-unused-vars
async create(data: AriesAgentData, params?: Params): Promise<any> {
switch (data.service) {
case ServiceType.Connection:
if (data.action === ConnectionServiceAction.Accept_Connection_Request) {
return this.acceptConnectionRequest(data.data.connection_id);
} else if (
data.action === ConnectionServiceAction.Send_Connection_Ping
) {
return this.sendConnectionPing(data.data.connection_id);
}
default:
return new NotImplemented(
`The operation ${data.service}/${data.action} is not supported`,
);
}
}
private async acceptConnectionRequest(
connection_id: string,
): Promise<boolean> {
try {
const url = `${this.acaPyUtils.getAdminUrl()}/connections/${connection_id}/accept-request`;
logger.debug(
`Accept connection request for connection with id ${connection_id}`,
);
const response = await Axios.post(
url,
{},
this.acaPyUtils.getRequestConfig(),
);
return response.status === 200 ? true : false;
} catch (e) {
const error = e as AxiosError;
throw new AriesAgentError(
error.response?.statusText || error.message,
error.response?.status,
error.response?.data,
);
}
}
private async sendConnectionPing(connection_id: string): Promise<boolean> {
try {
logger.debug(`Ping connection with id ${connection_id}`);
const url = `${this.acaPyUtils.getAdminUrl()}/connections/${connection_id}/send-ping`;
const response = await Axios.post(
url,
{},
this.acaPyUtils.getRequestConfig(),
);
return response.status === 200 ? true : false;
} catch (e) {
const error = e as AxiosError;
throw new AriesAgentError(
error.response?.statusText || error.message,
error.response?.status,
error.response?.data,
);
}
}
}
import { HookContext } from '@feathersjs/feathers';
import { disallow } from 'feathers-hooks-common';
export default {
before: {
all: [disallow('external')],
find: [],
get: [],
create: [],
update: [],
patch: [],
remove: [],
},
after: {
all: [disallow('external')],
find: [],
get: [],
create: [],
update: [],
patch: [],
remove: [],
},
error: {
all: [
disallow('external'),
async (context: HookContext): Promise<HookContext> => {
console.error(
`Error in ${context.path} calling ${context.method} method`,
context.error,
);
return context;
},
],
find: [],
get: [],
create: [],
update: [],
patch: [],
remove: [],
},
};
// Initializes the `aries-agent` service on path `/aries-agent`
import { ServiceAddons } from '@feathersjs/feathers';
import { Application } from '../../declarations';
import { AriesAgent } from './aries-agent.class';
import hooks from './aries-agent.hooks';
// Add this service to the service type index
declare module '../../declarations' {
interface ServiceTypes {
'aries-agent': AriesAgent & ServiceAddons<any>;
}
}
export default function (app: Application): void {
const options = {};
// Initialize our service with any options it requires
app.use('/aries-agent', new AriesAgent(options, app));
// Get our initialized service so that we can register hooks
const service = app.service('aries-agent');
service.hooks(hooks);
}
import { Application } from '../declarations';
import ariesAgent from './aries-agent/aries-agent.service';
import webhooks from './webhooks/webhooks.service';
// Don't remove this comment. It's needed to format import lines nicely.
export default function (app: Application): void {
app.configure(ariesAgent);
app.configure(webhooks);
}
import { NotImplemented } from '@feathersjs/errors';
import { Params, ServiceMethods } from '@feathersjs/feathers';
import { Application } from '../../declarations';
import {
ConnectionState,
ConnectionServiceAction,
ServiceType,
WebhookTopic,
} from '../../models/enums';
import { AriesAgentData } from '../aries-agent/aries-agent.class';
interface Data {
state: ConnectionState;
connection_id?: string;
transaction_id?: string;
}
interface ServiceOptions {}
export class Webhooks implements Partial<ServiceMethods<Data>> {
app: Application;
options: ServiceOptions;
constructor(options: ServiceOptions = {}, app: Application) {
this.options = options;
this.app = app;
}
// eslint-disable-next-line @typescript-eslint/no-unused-vars
async create(data: Data, params?: Params): Promise<any> {
const topic = params?.route?.topic;
const state = data?.state;
console.log('Received webhook:', topic, state);
switch (topic) {
case WebhookTopic.Connections:
if (state === ConnectionState.Request) {
// auto-accept connection requests
await this.app.service('aries-agent').create({
service: ServiceType.Connection,
action: ConnectionServiceAction.Accept_Connection_Request,
data: {
connection_id: data.connection_id,
},
} as AriesAgentData);
} else if (state === ConnectionState.Response) {
// auto-ping completes connections
await this.app.service('aries-agent').create({
service: ServiceType.Connection,
action: ConnectionServiceAction.Send_Connection_Ping,
data: {
connection_id: data.connection_id,
},
} as AriesAgentData);
} else if (state === ConnectionState.Completed) {
// No-op I think ...
}
return { result: 'Success' };
default:
return new NotImplemented(`Webhook ${topic} is not supported`);
}
}
}
export default {
before: {
all: [],
find: [],
get: [],
create: [],
update: [],
patch: [],
remove: [],
},
after: {
all: [],
find: [],
get: [],
create: [],
update: [],
patch: [],
remove: [],
},
error: {
all: [],
find: [],
get: [],
create: [],
update: [],
patch: [],
remove: [],
},
};
// Initializes the `webhooks` service on path `/webhooks/topic/:topic`
import { ServiceAddons } from '@feathersjs/feathers';
import { Application } from '../../declarations';
import { Webhooks } from './webhooks.class';
import hooks from './webhooks.hooks';
// Add this service to the service type index
declare module '../../declarations' {
interface ServiceTypes {
'webhooks/topic/:topic': Webhooks & ServiceAddons<any>;
}
}
export default function (app: Application): void {
const options = {
paginate: app.get('paginate'),
};
// Initialize our service with any options it requires
app.use('/webhooks/topic/:topic', new Webhooks(options, app));
// Get our initialized service so that we can register hooks
const service = app.service('webhooks/topic/:topic');
service.hooks(hooks);
}
import { Application } from '@feathersjs/express';
import Axios, { AxiosRequestConfig } from 'axios';
import logger from '../logger';
import { UndefinedAppError } from '../models/errors';
import { sleep } from './sleep';
export class AcaPyUtils {
static instance: AcaPyUtils;
app: Application;
private constructor(app: Application) {
this.app = app;
}
static getInstance(app?: Application): AcaPyUtils {
if (!this.instance) {
if (!app) {
throw new UndefinedAppError(
'Error creating a new instance of [AcaPyUtils]: no app was provided',
);
}
this.instance = new AcaPyUtils(app);
logger.debug('Created new instance of [AcaPyUtils]');
}
return this.instance;
}
getRequestConfig(): AxiosRequestConfig {
return {
headers: this.app.get('agent').headers,
} as AxiosRequestConfig;
}
getAdminUrl(): string {
return this.app.get('agent').adminUrl;
}
async init(): Promise<any> {
// wait for the agent to be ready
while (!(await this.isAgentReady())) {
logger.debug('Agent not ready, retrying in 3s...');
await sleep(3000);
}
return Promise.resolve({});
}
async isAgentReady(): Promise<boolean> {
const url = `${this.app.get('agent').adminUrl}/status/ready`;
let result;
try {
const response = await Axios.get(url, this.getRequestConfig());
result = response.status === 200 ? true : false;
} catch (error) {
result = false;
}
return Promise.resolve(result);
}
}
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment