mirror of
https://github.com/logto-io/logto.git
synced 2024-12-16 20:26:19 -05:00
feat(connector): create SMSAero connector (#4096)
* feat(connector): create SMSAero connector * fix(connector): smsaero connector code review fixes * chore: add changeset for sms aero --------- Co-authored-by: wangsijie <wangsijie94@gmail.com>
This commit is contained in:
parent
97fc519ac1
commit
3e72b6d56d
10 changed files with 436 additions and 2 deletions
5
.changeset/dull-starfishes-scream.md
Normal file
5
.changeset/dull-starfishes-scream.md
Normal file
|
@ -0,0 +1,5 @@
|
|||
---
|
||||
"@logto/connector-smsaero": minor
|
||||
---
|
||||
|
||||
Add SMSAero connector
|
64
packages/connectors/connector-smsaero/README.md
Normal file
64
packages/connectors/connector-smsaero/README.md
Normal file
|
@ -0,0 +1,64 @@
|
|||
# SMSAero short message service connector
|
||||
|
||||
The official Logto connector for SMSAero short message service.
|
||||
|
||||
**Table of contents**
|
||||
|
||||
- [SMSAero short message service connector](#smsaero-short-message-service-connector)
|
||||
- [Register account](#register-account)
|
||||
- [Set up senders' phone numbers](#set-up-senders-phone-numbers)
|
||||
- [Get account credentials](#get-account-credentials)
|
||||
- [Compose the connector JSON](#compose-the-connector-json)
|
||||
- [Test SMSAero connector](#test-smsaero-connector)
|
||||
- [Config types](#config-types)
|
||||
- [Reference](#reference)
|
||||
|
||||
## Register account
|
||||
|
||||
Create a new account on [SMSAero](https://smsaero.ru/). (Jump to the next step if you already have one.)
|
||||
|
||||
## Get account credentials
|
||||
|
||||
We will need the API credentials to make the connector work. Let's begin from
|
||||
the [API and SMPP](https://smsaero.ru/cabinet/settings/apikey/).
|
||||
|
||||
Copy "API-key" or generate new one.
|
||||
|
||||
## Compose the connector JSON
|
||||
|
||||
Fill out the _email_, _apiKey_ and _senderName_ fields with your email, API key and sender name.
|
||||
|
||||
You can fill sender name with "SMSAero" to use default sender name provided by SMSAero.
|
||||
|
||||
You can add multiple SMS connector templates for different cases. Here is an example of adding a single template:
|
||||
|
||||
- Fill out the `content` field with arbitrary string-typed contents. Do not forget to leave `{{code}}` placeholder for
|
||||
random verification code.
|
||||
- Fill out the `usageType` field with either `Register`, `SignIn`, `ForgotPassword`, `Generic` for different use cases.
|
||||
In order to enable full user flows, templates with usageType `Register`, `SignIn`, `ForgotPassword` and `Generic` are
|
||||
required.
|
||||
|
||||
### Test SMSAero connector
|
||||
|
||||
You can enter a phone number and click on "Send" to see whether the settings can work before "Save and Done".
|
||||
|
||||
That's it. Don't forget
|
||||
to [Enable connector in sign-in experience](https://docs.logto.io/docs/tutorials/get-started/passwordless-sign-in-by-adding-connectors#enable-sms-or-email-passwordless-sign-in).
|
||||
|
||||
### Config types
|
||||
|
||||
| Name | Type |
|
||||
|------------|-------------|
|
||||
| email | string |
|
||||
| apiKey | string |
|
||||
| senderName | string |
|
||||
| templates | Templates[] |
|
||||
|
||||
| Template Properties | Type | Enum values |
|
||||
|---------------------|-------------|---------------------------------------------------------|
|
||||
| content | string | N/A |
|
||||
| usageType | enum string | 'Register' \| 'SignIn' \| 'ForgotPassword' \| 'Generic' |
|
||||
|
||||
## Reference
|
||||
|
||||
- [SMSAero API Documentation](https://smsaero.ru/integration/documentation/api/)
|
8
packages/connectors/connector-smsaero/logo.svg
Normal file
8
packages/connectors/connector-smsaero/logo.svg
Normal file
|
@ -0,0 +1,8 @@
|
|||
<?xml version="1.0" standalone="no"?>
|
||||
<!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.0//EN" "http://www.w3.org/TR/2001/REC-SVG-20010904/DTD/svg10.dtd">
|
||||
<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" id="body_1" width="192" height="192">
|
||||
|
||||
<g transform="matrix(1.3333334 0 0 1.3333334 0 0)">
|
||||
<image x="0" y="0" xlink:href="" preserveAspectRatio="none" width="144" height="144"/>
|
||||
</g>
|
||||
</svg>
|
After Width: | Height: | Size: 4.5 KiB |
51
packages/connectors/connector-smsaero/package.json
Normal file
51
packages/connectors/connector-smsaero/package.json
Normal file
|
@ -0,0 +1,51 @@
|
|||
{
|
||||
"name": "@logto/connector-smsaero",
|
||||
"version": "1.0.0",
|
||||
"description": "SMSAero connector implementation.",
|
||||
"author": "Danil Tankov <danil.tankoff@yandex.ru>",
|
||||
"dependencies": {
|
||||
"@logto/connector-kit": "workspace:^1.1.0"
|
||||
},
|
||||
"main": "./lib/index.js",
|
||||
"module": "./lib/index.js",
|
||||
"exports": "./lib/index.js",
|
||||
"license": "MPL-2.0",
|
||||
"type": "module",
|
||||
"files": [
|
||||
"lib",
|
||||
"docs",
|
||||
"logo.svg",
|
||||
"logo-dark.svg"
|
||||
],
|
||||
"scripts": {
|
||||
"precommit": "lint-staged",
|
||||
"build:test": "rm -rf lib/ && tsc -p tsconfig.test.json --sourcemap",
|
||||
"build": "rm -rf lib/ && tsc -p tsconfig.build.json --noEmit && rollup -c",
|
||||
"dev": "tsc -p tsconfig.build.json --watch --preserveWatchOutput --incremental",
|
||||
"lint": "eslint --ext .ts src",
|
||||
"lint:report": "pnpm lint --format json --output-file report.json",
|
||||
"test:only": "NODE_OPTIONS=--experimental-vm-modules jest",
|
||||
"test": "pnpm build:test && pnpm test:only",
|
||||
"test:ci": "pnpm test:only --silent --coverage",
|
||||
"prepublishOnly": "pnpm build"
|
||||
},
|
||||
"engines": {
|
||||
"node": "^18.12.0"
|
||||
},
|
||||
"eslintConfig": {
|
||||
"extends": "@silverhand",
|
||||
"settings": {
|
||||
"import/core-modules": [
|
||||
"@silverhand/essentials",
|
||||
"got",
|
||||
"nock",
|
||||
"snakecase-keys",
|
||||
"zod"
|
||||
]
|
||||
}
|
||||
},
|
||||
"prettier": "@silverhand/eslint-config/.prettierrc",
|
||||
"publishConfig": {
|
||||
"access": "public"
|
||||
}
|
||||
}
|
70
packages/connectors/connector-smsaero/src/constant.ts
Normal file
70
packages/connectors/connector-smsaero/src/constant.ts
Normal file
|
@ -0,0 +1,70 @@
|
|||
import type { ConnectorMetadata } from '@logto/connector-kit';
|
||||
import { ConnectorConfigFormItemType } from '@logto/connector-kit';
|
||||
|
||||
export const endpoint = `https://gate.smsaero.ru/v2/sms/send`;
|
||||
|
||||
export const defaultMetadata: ConnectorMetadata = {
|
||||
id: 'smsaero-short-message-service',
|
||||
target: 'smsaero-sms',
|
||||
platform: null,
|
||||
name: {
|
||||
en: 'SMS Aero service',
|
||||
},
|
||||
logo: './logo.svg',
|
||||
logoDark: null,
|
||||
description: {
|
||||
en: 'SMS Aero offers users to use SMS-mailing in 5 minutes without viewing the contract. Developers are offered a convenient API with accessible classes and 24x7 chat support.',
|
||||
},
|
||||
readme: './README.md',
|
||||
formItems: [
|
||||
{
|
||||
key: 'email',
|
||||
label: 'Email',
|
||||
type: ConnectorConfigFormItemType.Text,
|
||||
required: true,
|
||||
placeholder: '<account-email>',
|
||||
},
|
||||
{
|
||||
key: 'apiKey',
|
||||
label: 'API Key',
|
||||
type: ConnectorConfigFormItemType.Text,
|
||||
required: true,
|
||||
placeholder: '<api-key>',
|
||||
},
|
||||
{
|
||||
key: 'senderName',
|
||||
label: 'Sender Name',
|
||||
type: ConnectorConfigFormItemType.Text,
|
||||
required: true,
|
||||
placeholder: 'SMSAero',
|
||||
},
|
||||
{
|
||||
key: 'templates',
|
||||
label: 'Templates',
|
||||
type: ConnectorConfigFormItemType.Json,
|
||||
required: true,
|
||||
defaultValue: [
|
||||
{
|
||||
usageType: 'SignIn',
|
||||
content:
|
||||
'Your Logto sign-in verification code is {{code}}. The code will remain active for 10 minutes.',
|
||||
},
|
||||
{
|
||||
usageType: 'Register',
|
||||
content:
|
||||
'Your Logto sign-up verification code is {{code}}. The code will remain active for 10 minutes.',
|
||||
},
|
||||
{
|
||||
usageType: 'ForgotPassword',
|
||||
content:
|
||||
'Your Logto password change verification code is {{code}}. The code will remain active for 10 minutes.',
|
||||
},
|
||||
{
|
||||
usageType: 'Generic',
|
||||
content:
|
||||
'Your Logto verification code is {{code}}. The code will remain active for 10 minutes.',
|
||||
},
|
||||
],
|
||||
},
|
||||
],
|
||||
};
|
12
packages/connectors/connector-smsaero/src/index.test.ts
Normal file
12
packages/connectors/connector-smsaero/src/index.test.ts
Normal file
|
@ -0,0 +1,12 @@
|
|||
import createConnector from './index.js';
|
||||
import { mockedConfig } from './mock.js';
|
||||
|
||||
const { jest } = import.meta;
|
||||
|
||||
const getConfig = jest.fn().mockResolvedValue(mockedConfig);
|
||||
|
||||
describe('SMSAero SMS connector', () => {
|
||||
it('init without throwing errors', async () => {
|
||||
await expect(createConnector({ getConfig })).resolves.not.toThrow();
|
||||
});
|
||||
});
|
85
packages/connectors/connector-smsaero/src/index.ts
Normal file
85
packages/connectors/connector-smsaero/src/index.ts
Normal file
|
@ -0,0 +1,85 @@
|
|||
import { assert } from '@silverhand/essentials';
|
||||
import { got, HTTPError } from 'got';
|
||||
|
||||
import type {
|
||||
CreateConnector,
|
||||
GetConnectorConfig,
|
||||
SendMessageFunction,
|
||||
SmsConnector,
|
||||
} from '@logto/connector-kit';
|
||||
import {
|
||||
ConnectorError,
|
||||
ConnectorErrorCodes,
|
||||
ConnectorType,
|
||||
validateConfig,
|
||||
} from '@logto/connector-kit';
|
||||
|
||||
import { defaultMetadata, endpoint } from './constant.js';
|
||||
import type { PublicParameters, SmsAeroConfig } from './types.js';
|
||||
import { smsAeroConfigGuard } from './types.js';
|
||||
|
||||
function sendMessage(getConfig: GetConnectorConfig): SendMessageFunction {
|
||||
return async (data, inputConfig) => {
|
||||
const { to, type, payload } = data;
|
||||
|
||||
const config = inputConfig ?? (await getConfig(defaultMetadata.id));
|
||||
validateConfig<SmsAeroConfig>(config, smsAeroConfigGuard);
|
||||
|
||||
const { email, apiKey, senderName, templates } = config;
|
||||
|
||||
const template = templates.find((template) => template.usageType === type);
|
||||
|
||||
assert(
|
||||
template,
|
||||
new ConnectorError(
|
||||
ConnectorErrorCodes.TemplateNotFound,
|
||||
`Cannot find template for type: ${type}`
|
||||
)
|
||||
);
|
||||
|
||||
const parameters: PublicParameters = {
|
||||
number: to,
|
||||
sign: senderName,
|
||||
text: template.content.replace(/{{code}}/g, payload.code),
|
||||
};
|
||||
|
||||
const auth = Buffer.from(`${email}:${apiKey}`).toString('base64');
|
||||
|
||||
try {
|
||||
return await got.post(endpoint, {
|
||||
headers: {
|
||||
Authorization: `Basic ${auth}`,
|
||||
},
|
||||
json: parameters,
|
||||
});
|
||||
} catch (error: unknown) {
|
||||
if (error instanceof HTTPError) {
|
||||
const {
|
||||
response: { body: rawBody },
|
||||
} = error;
|
||||
assert(
|
||||
typeof rawBody === 'string',
|
||||
new ConnectorError(
|
||||
ConnectorErrorCodes.InvalidResponse,
|
||||
`Invalid response raw body type: ${typeof rawBody}`
|
||||
)
|
||||
);
|
||||
|
||||
throw new ConnectorError(ConnectorErrorCodes.General, rawBody);
|
||||
}
|
||||
|
||||
throw error;
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
const createSmsAeroConnector: CreateConnector<SmsConnector> = async ({ getConfig }) => {
|
||||
return {
|
||||
metadata: defaultMetadata,
|
||||
type: ConnectorType.Sms,
|
||||
configGuard: smsAeroConfigGuard,
|
||||
sendMessage: sendMessage(getConfig),
|
||||
};
|
||||
};
|
||||
|
||||
export default createSmsAeroConnector;
|
17
packages/connectors/connector-smsaero/src/mock.ts
Normal file
17
packages/connectors/connector-smsaero/src/mock.ts
Normal file
|
@ -0,0 +1,17 @@
|
|||
import type { SmsAeroConfig } from './types.js';
|
||||
|
||||
const mockedEmail = 'test@test.com';
|
||||
const mockedApiKey = 'api-key';
|
||||
const mockedSenderName = 'sender-name';
|
||||
|
||||
export const mockedConfig: SmsAeroConfig = {
|
||||
email: mockedEmail,
|
||||
apiKey: mockedApiKey,
|
||||
senderName: mockedSenderName,
|
||||
templates: [
|
||||
{
|
||||
usageType: 'Test',
|
||||
content: 'This is for testing purposes only. Your verification code is {{code}}.',
|
||||
},
|
||||
],
|
||||
};
|
43
packages/connectors/connector-smsaero/src/types.ts
Normal file
43
packages/connectors/connector-smsaero/src/types.ts
Normal file
|
@ -0,0 +1,43 @@
|
|||
import { z } from 'zod';
|
||||
|
||||
/**
|
||||
* @doc https://smsaero.ru/integration/documentation/api/
|
||||
*/
|
||||
|
||||
export type PublicParameters = {
|
||||
number: string;
|
||||
sign: string;
|
||||
text: string;
|
||||
};
|
||||
|
||||
/**
|
||||
* UsageType here is used to specify the use case of the template, can be either
|
||||
* 'Register', 'SignIn', 'ForgotPassword', 'Generic' or 'Test'.
|
||||
*/
|
||||
const requiredTemplateUsageTypes = ['Register', 'SignIn', 'ForgotPassword', 'Generic'];
|
||||
|
||||
const templateGuard = z.object({
|
||||
usageType: z.string(),
|
||||
content: z.string(),
|
||||
});
|
||||
|
||||
export const smsAeroConfigGuard = z.object({
|
||||
email: z.string().email(),
|
||||
apiKey: z.string(),
|
||||
senderName: z.string(),
|
||||
templates: z.array(templateGuard).refine(
|
||||
(templates) =>
|
||||
requiredTemplateUsageTypes.every((requiredType) =>
|
||||
templates.map((template) => template.usageType).includes(requiredType)
|
||||
),
|
||||
(templates) => ({
|
||||
message: `Template with UsageType (${requiredTemplateUsageTypes
|
||||
.filter(
|
||||
(requiredType) => !templates.map((template) => template.usageType).includes(requiredType)
|
||||
)
|
||||
.join(', ')}) should be provided!`,
|
||||
})
|
||||
),
|
||||
});
|
||||
|
||||
export type SmsAeroConfig = z.infer<typeof smsAeroConfigGuard>;
|
|
@ -2263,6 +2263,85 @@ importers:
|
|||
specifier: ^5.0.0
|
||||
version: 5.0.2
|
||||
|
||||
packages/connectors/connector-smsaero:
|
||||
dependencies:
|
||||
'@logto/connector-kit':
|
||||
specifier: workspace:^1.1.0
|
||||
version: link:../../toolkit/connector-kit
|
||||
'@silverhand/essentials':
|
||||
specifier: ^2.5.0
|
||||
version: 2.5.0
|
||||
got:
|
||||
specifier: ^13.0.0
|
||||
version: 13.0.0
|
||||
snakecase-keys:
|
||||
specifier: ^5.4.4
|
||||
version: 5.4.4
|
||||
zod:
|
||||
specifier: ^3.20.2
|
||||
version: 3.20.2
|
||||
devDependencies:
|
||||
'@jest/types':
|
||||
specifier: ^29.5.0
|
||||
version: 29.5.0
|
||||
'@rollup/plugin-commonjs':
|
||||
specifier: ^25.0.0
|
||||
version: 25.0.0(rollup@3.8.0)
|
||||
'@rollup/plugin-json':
|
||||
specifier: ^6.0.0
|
||||
version: 6.0.0(rollup@3.8.0)
|
||||
'@rollup/plugin-node-resolve':
|
||||
specifier: ^15.0.1
|
||||
version: 15.0.1(rollup@3.8.0)
|
||||
'@rollup/plugin-typescript':
|
||||
specifier: ^11.0.0
|
||||
version: 11.0.0(rollup@3.8.0)(typescript@5.0.2)
|
||||
'@silverhand/eslint-config':
|
||||
specifier: 3.0.1
|
||||
version: 3.0.1(eslint@8.34.0)(prettier@2.8.4)(typescript@5.0.2)
|
||||
'@silverhand/ts-config':
|
||||
specifier: 3.0.0
|
||||
version: 3.0.0(typescript@5.0.2)
|
||||
'@types/jest':
|
||||
specifier: ^29.4.0
|
||||
version: 29.4.0
|
||||
'@types/node':
|
||||
specifier: ^18.11.18
|
||||
version: 18.11.18
|
||||
'@types/supertest':
|
||||
specifier: ^2.0.11
|
||||
version: 2.0.11
|
||||
eslint:
|
||||
specifier: ^8.34.0
|
||||
version: 8.34.0
|
||||
jest:
|
||||
specifier: ^29.5.0
|
||||
version: 29.5.0(@types/node@18.11.18)
|
||||
jest-matcher-specific-error:
|
||||
specifier: ^1.0.0
|
||||
version: 1.0.0
|
||||
lint-staged:
|
||||
specifier: ^13.0.0
|
||||
version: 13.0.0
|
||||
nock:
|
||||
specifier: ^13.2.2
|
||||
version: 13.2.2
|
||||
prettier:
|
||||
specifier: ^2.8.2
|
||||
version: 2.8.4
|
||||
rollup:
|
||||
specifier: ^3.8.0
|
||||
version: 3.8.0
|
||||
rollup-plugin-summary:
|
||||
specifier: ^2.0.0
|
||||
version: 2.0.0(rollup@3.8.0)
|
||||
supertest:
|
||||
specifier: ^6.2.2
|
||||
version: 6.2.2
|
||||
typescript:
|
||||
specifier: ^5.0.0
|
||||
version: 5.0.2
|
||||
|
||||
packages/connectors/connector-smtp:
|
||||
dependencies:
|
||||
'@logto/connector-kit':
|
||||
|
@ -10745,7 +10824,7 @@ packages:
|
|||
dev: false
|
||||
|
||||
/concat-map@0.0.1:
|
||||
resolution: {integrity: sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=}
|
||||
resolution: {integrity: sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==}
|
||||
dev: true
|
||||
|
||||
/concat-stream@2.0.0:
|
||||
|
@ -11328,7 +11407,7 @@ packages:
|
|||
dev: true
|
||||
|
||||
/dezalgo@1.0.3:
|
||||
resolution: {integrity: sha1-f3Qt4Gb8dIvI24IFad3c5Jvw1FY=}
|
||||
resolution: {integrity: sha512-K7i4zNfT2kgQz3GylDw40ot9GAE47sFZ9EXHFSPP6zONLgH6kWXE0KWJchkbQJLBkRazq4APwZ4OwiFFlT95OQ==}
|
||||
dependencies:
|
||||
asap: 2.0.6
|
||||
wrappy: 1.0.2
|
||||
|
|
Loading…
Reference in a new issue