Compare commits

..

19 commits

Author SHA1 Message Date
qpismont
765af2700d upgrade deps
All checks were successful
ci/woodpecker/push/build Pipeline was successful
ci/woodpecker/push/lint Pipeline was successful
ci/woodpecker/push/test/1 Pipeline was successful
ci/woodpecker/push/test/2 Pipeline was successful
ci/woodpecker/push/test/3 Pipeline was successful
2024-07-31 20:47:07 +02:00
da965206f6 Merge pull request 'upgrade to bun 1.1.10 + deps' (#8) from upgrade-bun1-1-10 into main
All checks were successful
ci/woodpecker/push/build Pipeline was successful
ci/woodpecker/push/lint Pipeline was successful
ci/woodpecker/push/test/1 Pipeline was successful
ci/woodpecker/push/test/2 Pipeline was successful
ci/woodpecker/push/test/3 Pipeline was successful
Reviewed-on: #8
2024-05-27 20:01:47 +02:00
qpismont
226bf76e1c fix biome ci
All checks were successful
ci/woodpecker/push/build Pipeline was successful
ci/woodpecker/push/lint Pipeline was successful
ci/woodpecker/push/test/1 Pipeline was successful
ci/woodpecker/push/test/2 Pipeline was successful
ci/woodpecker/push/test/3 Pipeline was successful
2024-05-27 19:59:18 +02:00
qpismont
2a6fa5195a upgrade to bun 1.1.10 + deps
Some checks failed
ci/woodpecker/push/build Pipeline was successful
ci/woodpecker/push/lint Pipeline failed
ci/woodpecker/push/test/1 Pipeline was successful
ci/woodpecker/push/test/2 Pipeline was successful
ci/woodpecker/push/test/3 Pipeline was successful
2024-05-27 18:54:16 +02:00
qpismont
330e47d477 bump version
All checks were successful
ci/woodpecker/push/build Pipeline was successful
ci/woodpecker/push/lint Pipeline was successful
ci/woodpecker/push/test/1 Pipeline was successful
ci/woodpecker/push/test/2 Pipeline was successful
ci/woodpecker/push/test/3 Pipeline was successful
ci/woodpecker/push/test/4 Pipeline was successful
2024-04-10 19:53:09 +02:00
354405a253 Merge pull request 'add srv params in handler + upgrade deps' (#7) from srv-in-handler into main
Some checks are pending
ci/woodpecker/push/test/2 Pipeline is pending
ci/woodpecker/push/test/3 Pipeline is pending
ci/woodpecker/push/test/4 Pipeline is pending
ci/woodpecker/push/build Pipeline was successful
ci/woodpecker/push/lint Pipeline was successful
ci/woodpecker/push/test/1 Pipeline was successful
Reviewed-on: #7
2024-04-10 19:52:40 +02:00
qpismont
fb7cd0bf34 fix package.json
All checks were successful
ci/woodpecker/push/build Pipeline was successful
ci/woodpecker/push/lint Pipeline was successful
ci/woodpecker/push/test/1 Pipeline was successful
ci/woodpecker/push/test/2 Pipeline was successful
ci/woodpecker/push/test/3 Pipeline was successful
ci/woodpecker/push/test/4 Pipeline was successful
2024-04-10 19:48:26 +02:00
qpismont
ab29b28125 add srv params in handler + upgrade deps
Some checks failed
ci/woodpecker/push/build Pipeline was successful
ci/woodpecker/push/lint Pipeline failed
ci/woodpecker/push/test/1 Pipeline was successful
ci/woodpecker/push/test/2 Pipeline was successful
ci/woodpecker/push/test/3 Pipeline was successful
ci/woodpecker/push/test/4 Pipeline was successful
2024-04-10 19:46:07 +02:00
bf7037eb15 Merge pull request 'upgrade to bun 1.1.0' (#6) from bun1.1 into main
All checks were successful
ci/woodpecker/push/build Pipeline was successful
ci/woodpecker/push/lint Pipeline was successful
ci/woodpecker/push/test Pipeline was successful
Reviewed-on: #6
2024-04-01 19:24:59 +02:00
qpismont
d936a34a2e update gitignore
All checks were successful
ci/woodpecker/push/build Pipeline was successful
ci/woodpecker/push/lint Pipeline was successful
ci/woodpecker/push/test Pipeline was successful
2024-04-01 19:24:01 +02:00
qpismont
aae70c59c0 upgrade to bun 1.1.0
All checks were successful
ci/woodpecker/push/build Pipeline was successful
ci/woodpecker/push/lint Pipeline was successful
ci/woodpecker/push/test Pipeline was successful
2024-04-01 19:14:53 +02:00
qpismont
3bdb12b2a9 bump version
All checks were successful
ci/woodpecker/push/build/1 Pipeline was successful
ci/woodpecker/push/build/2 Pipeline was successful
ci/woodpecker/push/build/3 Pipeline was successful
ci/woodpecker/push/build/4 Pipeline was successful
ci/woodpecker/push/lint Pipeline was successful
ci/woodpecker/push/test/1 Pipeline was successful
ci/woodpecker/push/test/2 Pipeline was successful
ci/woodpecker/push/test/3 Pipeline was successful
2024-03-20 20:47:27 +01:00
16eb073aaa Merge pull request '[retry] upgrade to bun 1.0.33 + upgrade deps' (#5) from bun1.0.33 into main
All checks were successful
ci/woodpecker/push/build/1 Pipeline was successful
ci/woodpecker/push/build/2 Pipeline was successful
ci/woodpecker/push/build/3 Pipeline was successful
ci/woodpecker/push/build/4 Pipeline was successful
ci/woodpecker/push/lint Pipeline was successful
ci/woodpecker/push/test/1 Pipeline was successful
ci/woodpecker/push/test/2 Pipeline was successful
ci/woodpecker/push/test/3 Pipeline was successful
Reviewed-on: #5
2024-03-20 20:41:47 +01:00
qpismont
487a48406f try woodpecker
All checks were successful
ci/woodpecker/push/build/1 Pipeline was successful
ci/woodpecker/push/build/2 Pipeline was successful
ci/woodpecker/push/build/3 Pipeline was successful
ci/woodpecker/push/build/4 Pipeline was successful
ci/woodpecker/push/lint Pipeline was successful
ci/woodpecker/push/test/1 Pipeline was successful
ci/woodpecker/push/test/2 Pipeline was successful
ci/woodpecker/push/test/3 Pipeline was successful
2024-03-20 20:38:10 +01:00
qpismont
96f8b57564 upgrade to bun 1.0.33 + upgrade deps 2024-03-20 20:21:35 +01:00
955b3ded7e Merge pull request 'bun bun bun' (#3) from switch-to-bun into main
All checks were successful
ci/woodpecker/push/build/1 Pipeline was successful
ci/woodpecker/push/build/2 Pipeline was successful
ci/woodpecker/push/lint Pipeline was successful
ci/woodpecker/push/test/1 Pipeline was successful
ci/woodpecker/push/test/2 Pipeline was successful
Reviewed-on: #3
2024-02-08 20:33:08 +01:00
qpismont
00d45d670b only ci for push event
All checks were successful
ci/woodpecker/push/build/1 Pipeline was successful
ci/woodpecker/push/build/2 Pipeline was successful
ci/woodpecker/push/lint Pipeline was successful
ci/woodpecker/push/test/1 Pipeline was successful
ci/woodpecker/push/test/2 Pipeline was successful
2024-02-08 20:21:58 +01:00
qpismont
d8c98adeff update to bun 1.0.26
All checks were successful
ci/woodpecker/push/build/1 Pipeline was successful
ci/woodpecker/push/build/2 Pipeline was successful
ci/woodpecker/push/lint Pipeline was successful
ci/woodpecker/push/test/1 Pipeline was successful
ci/woodpecker/push/test/2 Pipeline was successful
ci/woodpecker/pr/build/1 Pipeline was successful
ci/woodpecker/pr/build/2 Pipeline was successful
ci/woodpecker/pr/lint Pipeline was successful
ci/woodpecker/pr/test/1 Pipeline was successful
ci/woodpecker/pr/test/2 Pipeline was successful
2024-02-08 20:20:11 +01:00
qpismont
8283e05c02 bun bun bun
All checks were successful
ci/woodpecker/push/build Pipeline was successful
ci/woodpecker/push/lint Pipeline was successful
ci/woodpecker/push/test Pipeline was successful
ci/woodpecker/pr/build Pipeline was successful
ci/woodpecker/pr/lint Pipeline was successful
ci/woodpecker/pr/test Pipeline was successful
2024-01-30 21:50:19 +01:00
24 changed files with 568 additions and 428 deletions

177
.gitignore vendored
View file

@ -1,2 +1,177 @@
# Based on https://raw.githubusercontent.com/github/gitignore/main/Node.gitignore
# Logs
logs
_.log
npm-debug.log_
yarn-debug.log*
yarn-error.log*
lerna-debug.log*
.pnpm-debug.log*
# Caches
.cache
# Diagnostic reports (https://nodejs.org/api/report.html)
report.[0-9]_.[0-9]_.[0-9]_.[0-9]_.json
# Runtime data
pids
_.pid
_.seed
*.pid.lock
# Directory for instrumented libs generated by jscoverage/JSCover
lib-cov
# Coverage directory used by tools like istanbul
coverage
*.lcov
# nyc test coverage
.nyc_output
# Grunt intermediate storage (https://gruntjs.com/creating-plugins#storing-task-files)
.grunt
# Bower dependency directory (https://bower.io/)
bower_components
# node-waf configuration
.lock-wscript
# Compiled binary addons (https://nodejs.org/api/addons.html)
build/Release
# Dependency directories
node_modules/ node_modules/
npm/ jspm_packages/
# Snowpack dependency directory (https://snowpack.dev/)
web_modules/
# TypeScript cache
*.tsbuildinfo
# Optional npm cache directory
.npm
# Optional eslint cache
.eslintcache
# Optional stylelint cache
.stylelintcache
# Microbundle cache
.rpt2_cache/
.rts2_cache_cjs/
.rts2_cache_es/
.rts2_cache_umd/
# Optional REPL history
.node_repl_history
# Output of 'npm pack'
*.tgz
# Yarn Integrity file
.yarn-integrity
# dotenv environment variable files
.env
.env.development.local
.env.test.local
.env.production.local
.env.local
# parcel-bundler cache (https://parceljs.org/)
.parcel-cache
# Next.js build output
.next
out
# Nuxt.js build / generate output
.nuxt
dist
# Gatsby files
# Comment in the public line in if your project uses Gatsby and not Next.js
# https://nextjs.org/blog/next-9-1#public-directory-support
# public
# vuepress build output
.vuepress/dist
# vuepress v2.x temp and cache directory
.temp
# Docusaurus cache and generated files
.docusaurus
# Serverless directories
.serverless/
# FuseBox cache
.fusebox/
# DynamoDB Local files
.dynamodb/
# TernJS port file
.tern-port
# Stores VSCode versions used for testing VSCode extensions
.vscode-test
# yarn v2
.yarn/cache
.yarn/unplugged
.yarn/build-state.yml
.yarn/install-state.gz
.pnp.*
# IntelliJ based IDEs
.idea
# Finder (MacOS) folder config
.DS_Store
.npmrc

View file

@ -1,3 +1,8 @@
{ {
"editor.formatOnSave": true "editor.formatOnSave": true,
} "editor.defaultFormatter": "biomejs.biome",
"editor.codeActionsOnSave": {
"source.organizeImports.biome": "explicit"
},
"typescript.tsdk": "node_modules/typescript/lib"
}

View file

@ -1,16 +1,9 @@
when:
event: [push]
steps: steps:
build: build:
image: denoland/deno:${DENO_VERSION} image: oven/bun:1.1.10-slim
commands: commands:
- apt update - bun install
- apt install curl -y - bun run build
- curl -fsSL https://deb.nodesource.com/setup_lts.x | bash - && apt-get install -y nodejs
- deno task build
matrix:
DENO_VERSION:
- 1.39.2
- 1.39.3
- 1.39.4
- 1.40.0
- 1.40.1

View file

@ -1,5 +1,10 @@
when:
event: [push]
steps: steps:
lint: lint:
image: denoland/deno:1.40.1 image: oven/bun:1.1.10-slim
commands: commands:
- deno lint - bun install
- bun run ci

View file

@ -4,7 +4,7 @@ when:
steps: steps:
publish: publish:
image: denoland/deno:1.40.1 image: denoland/deno:1.39.2
commands: commands:
- apt update - apt update
- apt install curl -y - apt install curl -y

View file

@ -1,19 +1,21 @@
when:
event: [push]
steps: steps:
test: test:
image: denoland/deno:${DENO_VERSION} image: oven/bun:${BUN_VERSION}-slim
environment: environment:
- NATS_HOST=nats:4222 - NATS_HOST=nats:4222
commands: commands:
- deno task test - bun install
- bun run test
matrix: matrix:
DENO_VERSION: BUN_VERSION:
- 1.39.2 - 1.1.8
- 1.39.3 - 1.1.9
- 1.39.4 - 1.1.10
- 1.40.0
- 1.40.1
services: services:
nats: nats:
image: nats image: nats:2-alpine

20
biome.json Normal file
View file

@ -0,0 +1,20 @@
{
"$schema": "https://biomejs.dev/schemas/1.7.3/schema.json",
"organizeImports": {
"enabled": true
},
"linter": {
"enabled": true,
"rules": {
"recommended": true
}
},
"formatter": {
"enabled": true
},
"vcs": {
"enabled": true,
"clientKind": "git",
"useIgnoreFile": true
}
}

BIN
bun.lockb Executable file

Binary file not shown.

View file

@ -1,16 +0,0 @@
{
"imports": {
"nats": "npm:nats",
"zod": "npm:zod",
"std/": "https://deno.land/std@0.213.0/",
"dnt": "https://deno.land/x/dnt@0.39.0/mod.ts"
},
"tasks": {
"test": "deno test --allow-net --allow-env --parallel",
"build": "deno run -A scripts/build_npm.ts 0.1.0",
"publish": "cd npm && npm publish"
},
"lint": {
"include": ["src/"]
}
}

View file

@ -1,73 +0,0 @@
{
"version": "3",
"packages": {
"specifiers": {
"npm:nats": "npm:nats@2.18.0",
"npm:zod": "npm:zod@3.22.4"
},
"npm": {
"nats@2.18.0": {
"integrity": "sha512-zZF004ejzf67Za0Tva+xphxoxBMNc5IMLqbZ7Ho0j9TMuisjpo+qCd1EktXRCLNxmrZ8O6Tbm1dBsZYNF6yR1A==",
"dependencies": {
"nkeys.js": "nkeys.js@1.0.5"
}
},
"nkeys.js@1.0.5": {
"integrity": "sha512-u25YnRPHiGVsNzwyHnn+PT90sgAhnS8jUJ1nxmkHMFYCJ6+Ic0lv291w7uhRBpJVJ3PH2GWbYqA151lGCRrB5g==",
"dependencies": {
"tweetnacl": "tweetnacl@1.0.3"
}
},
"tweetnacl@1.0.3": {
"integrity": "sha512-6rt+RN7aOi1nGMyC4Xa5DdYiukl2UWCbcJft7YhxReBGQD7OAM8Pbxw6YMo4r2diNEA8FEmu32YOn9rhaiE5yw==",
"dependencies": {}
},
"zod@3.22.4": {
"integrity": "sha512-iC+8Io04lddc+mVqQ9AZ7OQ2MrUKGN+oIQyq1vemgt46jwCwLfhq7/pwnBnNXXXZb8VTVLKwp9EDkx+ryxIWmg==",
"dependencies": {}
}
}
},
"remote": {
"https://deno.land/std@0.211.0/assert/_constants.ts": "a271e8ef5a573f1df8e822a6eb9d09df064ad66a4390f21b3e31f820a38e0975",
"https://deno.land/std@0.211.0/assert/_diff.ts": "6a2d68f2c42d73a1e31818a4195f40598d672c7f02ac75c7f1b1e6789852c2bc",
"https://deno.land/std@0.211.0/assert/_format.ts": "0ba808961bf678437fb486b56405b6fefad2cf87b5809667c781ddee8c32aff4",
"https://deno.land/std@0.211.0/assert/assert.ts": "bec068b2fccdd434c138a555b19a2c2393b71dfaada02b7d568a01541e67cdc5",
"https://deno.land/std@0.211.0/assert/assert_almost_equals.ts": "648ea72678296a5ad86d3bbb66904335fa97de3133223f44ca4596b225cdcbef",
"https://deno.land/std@0.211.0/assert/assert_array_includes.ts": "dbb461c20681807a884ad84d873f9e4daead380859531b1e7f27fa4e8f8bf431",
"https://deno.land/std@0.211.0/assert/assert_equals.ts": "b3b33ae8a85ae22a0754c61a7486d4ae870e8938830a94f5cacecba3a9b0442a",
"https://deno.land/std@0.211.0/assert/assert_exists.ts": "24a7bf965e634f909242cd09fbaf38bde6b791128ece08e33ab08586a7cc55c9",
"https://deno.land/std@0.211.0/assert/assert_false.ts": "6f382568e5128c0f855e5f7dbda8624c1ed9af4fcc33ef4a9afeeedcdce99769",
"https://deno.land/std@0.211.0/assert/assert_greater.ts": "8dfcf082d2bcffcaab3bd0dab48d41e41c26266529567246de47bd6864936f6d",
"https://deno.land/std@0.211.0/assert/assert_greater_or_equal.ts": "9e02ef89f32563f539f7e66556930033418728847aefcca4e3806a735b5f122e",
"https://deno.land/std@0.211.0/assert/assert_instance_of.ts": "72dc1faff1e248692d873c89382fa1579dd7b53b56d52f37f9874a75b11ba444",
"https://deno.land/std@0.211.0/assert/assert_is_error.ts": "6596f2b5ba89ba2fe9b074f75e9318cda97a2381e59d476812e30077fbdb6ed2",
"https://deno.land/std@0.211.0/assert/assert_less.ts": "91a6fed705f9c39bbd683b62aa9dfc42547bc886c29f696997e681cafb886b16",
"https://deno.land/std@0.211.0/assert/assert_less_or_equal.ts": "7a3c2e554eb20aa6af9dd4a410e550bcee9e8a28102d51f5f40cb1b8d141e4e1",
"https://deno.land/std@0.211.0/assert/assert_match.ts": "ec2d9680ed3e7b9746ec57ec923a17eef6d476202f339ad91d22277d7f1d16e1",
"https://deno.land/std@0.211.0/assert/assert_not_equals.ts": "cb78bf9a4357d69673c87b634491bc6b840412c8b55efe472af9877ef6f0a29b",
"https://deno.land/std@0.211.0/assert/assert_not_instance_of.ts": "8f720d92d83775c40b2542a8d76c60c2d4aeddaf8713c8d11df8984af2604931",
"https://deno.land/std@0.211.0/assert/assert_not_match.ts": "b4b7c77f146963e2b673c1ce4846473703409eb93f5ab0eb60f6e6f8aeffe39f",
"https://deno.land/std@0.211.0/assert/assert_not_strict_equals.ts": "89ba25e1da5233404ac4c01651c088759b7977c51034eefc6050fe3fc2d10c46",
"https://deno.land/std@0.211.0/assert/assert_object_match.ts": "e85e5eef62a56ce364c3afdd27978ccab979288a3e772e6855c270a7b118fa49",
"https://deno.land/std@0.211.0/assert/assert_rejects.ts": "e9e0c8d9c3e164c7ac962c37b3be50577c5a2010db107ed272c4c1afb1269f54",
"https://deno.land/std@0.211.0/assert/assert_strict_equals.ts": "0425a98f70badccb151644c902384c12771a93e65f8ff610244b8147b03a2366",
"https://deno.land/std@0.211.0/assert/assert_string_includes.ts": "dfb072a890167146f8e5bdd6fde887ce4657098e9f71f12716ef37f35fb6f4a7",
"https://deno.land/std@0.211.0/assert/assert_throws.ts": "edddd86b39606c342164b49ad88dd39a26e72a26655e07545d172f164b617fa7",
"https://deno.land/std@0.211.0/assert/assertion_error.ts": "9f689a101ee586c4ce92f52fa7ddd362e86434ffdf1f848e45987dc7689976b8",
"https://deno.land/std@0.211.0/assert/equal.ts": "fae5e8a52a11d3ac694bbe1a53e13a7969e3f60791262312e91a3e741ae519e2",
"https://deno.land/std@0.211.0/assert/fail.ts": "f310e51992bac8e54f5fd8e44d098638434b2edb802383690e0d7a9be1979f1c",
"https://deno.land/std@0.211.0/assert/mod.ts": "325df8c0683ad83a873b9691aa66b812d6275fc9fec0b2d180ac68a2c5efed3b",
"https://deno.land/std@0.211.0/assert/unimplemented.ts": "47ca67d1c6dc53abd0bd729b71a31e0825fc452dbcd4fde4ca06789d5644e7fd",
"https://deno.land/std@0.211.0/assert/unreachable.ts": "38cfecb95d8b06906022d2f9474794fca4161a994f83354fd079cac9032b5145",
"https://deno.land/std@0.211.0/fmt/colors.ts": "be082d6a6bbb2980ae7b2bf8c23c6bb2811ba90a06a9bcb861344a71784c5a99",
"https://deno.land/std@0.211.0/testing/_test_suite.ts": "f10a8a6338b60c403f07a76f3f46bdc9f1e1a820c0a1decddeb2949f7a8a0546",
"https://deno.land/std@0.211.0/testing/bdd.ts": "3cbd17bd35f629a76ce63446238dfb4632240dd46b3b205027c45fa3dd67e554"
},
"workspace": {
"dependencies": [
"npm:nats",
"npm:zod"
]
}
}

4
mod.ts
View file

@ -1,4 +0,0 @@
export * from "./src/service.ts";
export * from "./src/adaptors/nats.ts";
export * from "./src/error.ts";
export type { Request, Response } from "./src/messages.ts";

26
package.json Normal file
View file

@ -0,0 +1,26 @@
{
"name": "@qpismont/nano-service",
"version": "0.0.3",
"module": "index.ts",
"type": "module",
"main": "./dist/index.js",
"types": "./dist/index.d.ts",
"files": ["/dist"],
"scripts": {
"ci": "biome ci . --error-on-warnings",
"test": "bun test",
"build": "tsc --project tsconfig.build.json"
},
"devDependencies": {
"@biomejs/biome": "1.3.3",
"@types/bun": "latest",
"typescript": "^5.5.4"
},
"peerDependencies": {
"typescript": "^5.0.0"
},
"dependencies": {
"nats": "^2.28.2",
"zod": "^3.23.8"
}
}

View file

@ -1,27 +0,0 @@
import { build, emptyDir } from "dnt";
await emptyDir("./npm");
await build({
entryPoints: ["./mod.ts"],
outDir: "./npm",
shims: {
deno: true,
},
test: false,
importMap: "deno.json",
package: {
name: "@qpismont/nano-service",
version: Deno.args[0],
description: "Your package.",
license: "MIT",
repository: {
"type": "git",
"url": "https://gitea.qpismont.fr/qpismont/nano-service",
},
},
postBuild() {
Deno.copyFileSync("LICENSE", "npm/LICENSE");
Deno.copyFileSync("README.md", "npm/README.md");
},
});

View file

@ -1,8 +1,8 @@
import { AdaptorSubscribeTypeFn } from "../types.ts"; import type { AdaptorSubscribeTypeFn } from "../types.ts";
export default interface Adaptor { export default interface Adaptor {
subscribe(subject: string, fn: AdaptorSubscribeTypeFn): void; subscribe(subject: string, fn: AdaptorSubscribeTypeFn): void;
request(subject: string, req: string): Promise<string>; request(subject: string, req: string): Promise<string>;
listen(serviceName: string): Promise<void>; listen(serviceName: string): Promise<void>;
stop(): Promise<void>; stop(): Promise<void>;
} }

View file

@ -1,57 +1,58 @@
import { AdaptorSubscribeTypeFn } from "../types.ts";
import Adaptor from "./adaptor.ts";
import { import {
Codec, type Codec,
connect, type ConnectionOptions,
ConnectionOptions, type NatsConnection,
NatsConnection, StringCodec,
StringCodec, type Subscription,
Subscription, connect,
} from "nats"; } from "nats";
import type { AdaptorSubscribeTypeFn } from "../types.ts";
import type Adaptor from "./adaptor.ts";
export default class NatsAdaptor implements Adaptor { export default class NatsAdaptor implements Adaptor {
private options: ConnectionOptions; private options: ConnectionOptions;
private nc?: NatsConnection; private nc?: NatsConnection;
private callbacks: Record<string, AdaptorSubscribeTypeFn>; private callbacks: Record<string, AdaptorSubscribeTypeFn>;
private sc: Codec<string>; private sc: Codec<string>;
private sub?: Subscription; private sub?: Subscription;
constructor(options: ConnectionOptions) { constructor(options: ConnectionOptions) {
this.options = options; this.options = options;
this.sc = StringCodec(); this.sc = StringCodec();
this.callbacks = {}; this.callbacks = {};
} }
async listen(serviceName: string) { async listen(serviceName: string) {
this.nc = await connect(this.options); this.nc = await connect(this.options);
this.sub = this.nc.subscribe(`${serviceName}.*`); this.sub = this.nc.subscribe(`${serviceName}.*`);
(async (sub: Subscription) => { (async (sub: Subscription) => {
for await (const msg of sub) { for await (const msg of sub) {
const res = await this.callbacks[msg.subject](this.sc.decode(msg.data)); const res = await this.callbacks[msg.subject](this.sc.decode(msg.data));
msg.respond(this.sc.encode(res)); msg.respond(this.sc.encode(res));
} }
})(this.sub); })(this.sub);
} }
subscribe(subject: string, fn: AdaptorSubscribeTypeFn) { subscribe(subject: string, fn: AdaptorSubscribeTypeFn) {
this.callbacks[subject] = fn; this.callbacks[subject] = fn;
} }
async request(subject: string, req: string): Promise<string> { async request(subject: string, req: string): Promise<string> {
if (this.nc) { if (this.nc) {
const msg = await this.nc.request(subject, this.sc.encode(req)); const msg = await this.nc.request(subject, this.sc.encode(req));
return this.sc.decode(msg.data); return this.sc.decode(msg.data);
} }
throw new Error("nats connection is not initialized"); throw new Error("nats connection is not initialized");
} }
async stop(): Promise<void> { async stop(): Promise<void> {
await this.nc?.drain(); this.sub?.unsubscribe();
} await this.nc?.drain();
}
} }

View file

@ -1,9 +1,9 @@
export class RequestError extends Error { export class RequestError extends Error {
public statusCode: number; public statusCode: number;
constructor(msg: string, statusCode: number) { constructor(msg: string, statusCode: number) {
super(msg); super(msg);
this.statusCode = statusCode; this.statusCode = statusCode;
} }
} }

4
src/index.ts Normal file
View file

@ -0,0 +1,4 @@
export * from "./service";
export * from "./adaptors/nats";
export * from "./error";
export type { Request, Response } from "./messages";

View file

@ -1,25 +1,25 @@
import { z } from "zod"; import { z } from "zod";
export interface Request<T> { export interface Request<T> {
service: string; service: string;
subject: string; subject: string;
data?: T; data?: T;
} }
export const InternalRequestSchema = z.object({ export const InternalRequestSchema = z.object({
from: z.string(), from: z.string(),
data: z.optional(z.any()), data: z.optional(z.any()),
}); });
export const InternalResponseSchema = z.object({ export const InternalResponseSchema = z.object({
data: z.optional(z.any()), data: z.optional(z.any()),
statusCode: z.number(), statusCode: z.number(),
}); });
export type InternalRequest = z.infer<typeof InternalRequestSchema>; export type InternalRequest = z.infer<typeof InternalRequestSchema>;
export type InternalResponse = z.infer<typeof InternalResponseSchema>; export type InternalResponse = z.infer<typeof InternalResponseSchema>;
export interface Response<T> { export interface Response<T> {
data?: T; data?: T;
statusCode: number; statusCode: number;
} }

View file

@ -1,166 +1,160 @@
import Adaptor from "./adaptors/adaptor.ts"; import type { z } from "zod";
import { RequestError } from "./error.ts"; import type Adaptor from "./adaptors/adaptor";
import { RequestError } from "./error";
import { import {
InternalRequest, type InternalRequest,
InternalRequestSchema, InternalRequestSchema,
InternalResponse, type InternalResponse,
InternalResponseSchema, InternalResponseSchema,
Request, type Request,
Response, type Response,
} from "./messages.ts"; } from "./messages";
import { RouteSubscribeTypeFn } from "./types.ts"; import type { RouteSubscribeTypeFn } from "./types";
import { z } from "zod";
export default class Service { export default class Service {
private name: string; private name: string;
private adaptors: Record<string, Adaptor>; private adaptors: Record<string, Adaptor>;
constructor(name: string) { constructor(name: string) {
this.name = name; this.name = name;
this.adaptors = {}; this.adaptors = {};
} }
public addAdaptor(name: string, adaptor: Adaptor) { public addAdaptor(name: string, adaptor: Adaptor) {
this.adaptors[name] = adaptor; this.adaptors[name] = adaptor;
} }
public subscribe<T extends z.ZodType<O>, O, U>( public subscribe<T extends z.ZodType<O>, O, U>(
adaptor: string, adaptor: string,
subject: string, subject: string,
fn: RouteSubscribeTypeFn<z.infer<T>, U>, fn: RouteSubscribeTypeFn<z.infer<T>, U>,
schema: T | undefined = undefined, schema?: T,
) { ) {
this.adaptors[adaptor].subscribe( this.adaptors[adaptor].subscribe(
`${this.name}.${subject}`, `${this.name}.${subject}`,
async (rawReq) => { async (rawReq) => {
const rawReqJson = JSON.parse(rawReq); const rawReqJson = JSON.parse(rawReq);
const internalRequestJson = InternalRequestSchema.safeParse(rawReqJson); const internalRequestJson = InternalRequestSchema.safeParse(rawReqJson);
if (!internalRequestJson.success) {
return JSON.stringify(
{
statusCode: 400,
data: "bad request structure",
} satisfies InternalResponse,
);
}
const internalRequest = internalRequestJson.data; if (!internalRequestJson.success) {
const req = { return JSON.stringify({
service: internalRequest.from, statusCode: 400,
subject: subject, data: "bad request structure",
} as Request<z.infer<T>>; } satisfies InternalResponse);
}
if (internalRequest.data && schema) { const internalRequest = internalRequestJson.data;
const validate = schema.safeParse(internalRequest.data); const req = {
if (!validate.success) { service: internalRequest.from,
return JSON.stringify( subject: subject,
{ } as Request<z.infer<T>>;
statusCode: 400,
data: validate.error,
} satisfies InternalResponse,
);
} else {
req.data = validate.data;
}
}
try { if (internalRequest.data && schema) {
const res = await fn(req); const validate = schema.safeParse(internalRequest.data);
const internalResponse = {
statusCode: res.statusCode,
data: res.data,
} satisfies InternalResponse;
return JSON.stringify(internalResponse); if (!validate.success) {
} catch (err) { return JSON.stringify({
if (err instanceof RequestError) { statusCode: 400,
return JSON.stringify( data: validate.error,
{ } satisfies InternalResponse);
statusCode: err.statusCode, }
data: err.message,
} satisfies InternalResponse,
);
} else {
return JSON.stringify(
{
statusCode: 500,
data: err?.message || "unknow error apend",
} satisfies InternalResponse,
);
}
}
},
);
}
public async request<T, O, U extends z.ZodType<O>>( req.data = validate.data;
adaptor: string, }
req: Request<T>,
schema?: z.ZodType<O>,
): Promise<Response<z.infer<U>>> {
if (!this.adaptors[adaptor]) {
throw new Error(`${adaptor} adaptor not exist`);
}
const internalRequest = { try {
from: this.name, const res = await fn(this, req);
data: req.data, const internalResponse = {
} satisfies InternalRequest; statusCode: res.statusCode,
const internalRequestJson = JSON.stringify(internalRequest); data: res.data,
} satisfies InternalResponse;
try { return JSON.stringify(internalResponse);
const rawRes = await this.adaptors[adaptor].request( } catch (err) {
`${req.service}.${req.subject}`, if (err instanceof RequestError) {
internalRequestJson, return JSON.stringify({
); statusCode: err.statusCode,
const rawResJson: unknown = JSON.parse(rawRes); data: err.message,
const internalResponseJson = InternalResponseSchema.safeParse(rawResJson); } satisfies InternalResponse);
if (!internalResponseJson.success) { }
throw new RequestError(internalResponseJson.error.toString(), 500);
}
const internalResponse = internalResponseJson.data; return JSON.stringify({
if ( statusCode: 500,
internalResponse.statusCode < 200 || internalResponse.statusCode >= 299 data: "unknow error apend",
) { } satisfies InternalResponse);
throw new RequestError( }
internalResponse.data, },
internalResponse.statusCode, );
); }
}
const res: Response<z.infer<U>> = {
statusCode: internalResponse.statusCode,
};
if (internalResponse.data && schema) {
const validate = schema.safeParse(internalResponse.data);
if (!validate.success) { public async request<T, O, U extends z.ZodType<O>>(
throw new RequestError(validate.error.message, 400); adaptor: string,
} else { req: Request<T>,
res.data = validate.data; schema?: z.ZodType<O>,
} ): Promise<Response<z.infer<U>>> {
} if (!this.adaptors[adaptor]) {
throw new Error(`${adaptor} adaptor not exist`);
}
return res; const internalRequest = {
} catch (err) { from: this.name,
if (err instanceof RequestError) { data: req.data,
throw err; } satisfies InternalRequest;
} const internalRequestJson = JSON.stringify(internalRequest);
throw new RequestError("unexpected error", 500); try {
} const rawRes = await this.adaptors[adaptor].request(
} `${req.service}.${req.subject}`,
internalRequestJson,
);
const rawResJson: unknown = JSON.parse(rawRes);
const internalResponseJson = InternalResponseSchema.safeParse(rawResJson);
if (!internalResponseJson.success) {
throw new RequestError(internalResponseJson.error.toString(), 500);
}
public async listen() { const internalResponse = internalResponseJson.data;
for (const index in this.adaptors) { if (
await this.adaptors[index].listen(this.name); internalResponse.statusCode < 200 ||
} internalResponse.statusCode >= 299
} ) {
throw new RequestError(
internalResponse.data,
internalResponse.statusCode,
);
}
const res: Response<z.infer<U>> = {
statusCode: internalResponse.statusCode,
};
if (internalResponse.data && schema) {
const validate = schema.safeParse(internalResponse.data);
public async stop() { if (!validate.success) {
for (const index in this.adaptors) { throw new RequestError(validate.error.message, 400);
await this.adaptors[index].stop(); }
} res.data = validate.data;
} }
return res;
} catch (err) {
if (err instanceof RequestError) {
throw err;
}
throw new RequestError("unexpected error", 500);
}
}
public async listen() {
for (const index in this.adaptors) {
await this.adaptors[index].listen(this.name);
}
}
public async stop() {
for (const index in this.adaptors) {
await this.adaptors[index].stop();
}
}
} }

View file

@ -1,6 +1,8 @@
import { Request, Response } from "./messages.ts"; import type { Request, Response } from "./messages.ts";
import type Service from "./service.ts";
export type AdaptorSubscribeTypeFn = (msg: string) => Promise<string>; export type AdaptorSubscribeTypeFn = (msg: string) => Promise<string>;
export type RouteSubscribeTypeFn<T, U> = ( export type RouteSubscribeTypeFn<T, U> = (
msg: Request<T>, srv: Service,
msg: Request<T>,
) => Promise<Response<U>>; ) => Promise<Response<U>>;

View file

@ -1,12 +1,12 @@
import { RequestError } from "../src/error.ts"; import { expect, test } from "bun:test";
import { assertEquals } from "std/assert/mod.ts"; import { RequestError } from "../src/error";
Deno.test("request error", () => { test("request error", () => {
const expectedMsg = "This is my err"; const expectedMsg = "This is my err";
const expectedStatusCode = 500; const expectedStatusCode = 500;
const err = new RequestError(expectedMsg, expectedStatusCode); const err = new RequestError(expectedMsg, expectedStatusCode);
assertEquals(err.message, expectedMsg); expect(err.message).toBe(expectedMsg);
assertEquals(err.statusCode, expectedStatusCode); expect(err.statusCode).toBe(expectedStatusCode);
}); });

View file

@ -1,92 +1,92 @@
import { afterEach, beforeEach, expect, test } from "bun:test";
import { z } from "zod"; import { z } from "zod";
import NatsAdaptor from "../src/adaptors/nats.ts"; import NatsAdaptor from "../src/adaptors/nats";
import Service from "../src/service.ts"; import { RequestError } from "../src/error";
import { assertEquals, assertRejects, assertThrows } from "std/assert/mod.ts"; import Service from "../src/service";
import { afterEach, beforeEach, it } from "std/testing/bdd.ts";
import { RequestError } from "../src/error.ts";
let srv!: Service; let srv!: Service;
const natsServer = Deno.env.get("NATS_HOST") || "127.0.0.1:4222"; const natsServer = Bun.env.NATS_HOST || "127.0.0.1:4222";
const serviceName = "test-service"; const serviceName = "test-service";
const adaptorName = "nats"; const adaptorName = "nats";
const subject = "test"; const subject = "test";
beforeEach(() => { beforeEach(() => {
srv = new Service(serviceName); srv = new Service(serviceName);
}); });
afterEach(async () => { afterEach(async () => {
await srv.stop(); await srv.stop();
}); });
it("request success", { test("request success", async () => {
sanitizeExit: false, const dataExpected = "hello success";
sanitizeOps: false, const statusCodeExpected = 200;
sanitizeResources: false,
}, async () => {
const dataExpected = "hello success";
const statusCodeExpected = 200;
srv.addAdaptor(adaptorName, new NatsAdaptor({ servers: [natsServer] })); srv.addAdaptor(adaptorName, new NatsAdaptor({ servers: [natsServer] }));
srv.subscribe(adaptorName, subject, async (msg) => { srv.subscribe(
return { data: msg.data, statusCode: statusCodeExpected }; adaptorName,
}, z.string()); subject,
async (srv, msg) => {
return { data: msg.data, statusCode: statusCodeExpected };
},
z.string(),
);
await srv.listen(); await srv.listen();
const res = await srv.request(adaptorName, { const res = await srv.request(
service: serviceName, adaptorName,
subject: subject, {
data: dataExpected, service: serviceName,
}, z.string()); subject: subject,
data: dataExpected,
},
z.string(),
);
assertEquals(res.data, dataExpected); expect(res.data).toBe(dataExpected);
assertEquals(res.statusCode, statusCodeExpected); expect(res.statusCode).toBe(statusCodeExpected);
}); });
it("request error", { test("request error", async () => {
sanitizeExit: false, const dataExpected = "hello error";
sanitizeOps: false, const statusCodeExpected = 500;
sanitizeResources: false,
}, async () => {
const dataExpected = "hello error";
const statusCodeExpected = 500;
srv.addAdaptor(adaptorName, new NatsAdaptor({ servers: [natsServer] })); srv.addAdaptor(adaptorName, new NatsAdaptor({ servers: [natsServer] }));
srv.subscribe(adaptorName, subject, async (msg) => { srv.subscribe(adaptorName, subject, async (srv, msg) => {
throw new RequestError("request error", 500); throw new RequestError("request error", 500);
}, z.string()); });
await srv.listen(); await srv.listen();
assertRejects(() => { expect(() => {
return srv.request(adaptorName, { return srv.request(
service: serviceName, adaptorName,
subject: subject, {
data: dataExpected, service: serviceName,
}, z.string()); subject: subject,
}); data: dataExpected,
},
z.string(),
);
}).toThrow();
}); });
it("request adaptor not found", { test("request adaptor not found", async () => {
sanitizeExit: false, const dataExpected = "hello success";
sanitizeOps: false, const statusCodeExpected = 200;
sanitizeResources: false,
}, async () => {
const dataExpected = "hello success";
const statusCodeExpected = 200;
srv.addAdaptor(adaptorName, new NatsAdaptor({ servers: [natsServer] })); srv.addAdaptor(adaptorName, new NatsAdaptor({ servers: [natsServer] }));
srv.subscribe(adaptorName, subject, async (msg) => { srv.subscribe(adaptorName, subject, async (srv, msg) => {
return { data: msg.data, statusCode: statusCodeExpected }; return { data: msg.data, statusCode: statusCodeExpected };
}); });
await srv.listen(); await srv.listen();
assertRejects(() => { expect(() => {
return srv.request("bad adaptor", { return srv.request("bad adaptor", {
service: serviceName, service: serviceName,
subject: subject, subject: subject,
data: dataExpected, data: dataExpected,
}); });
}); }).toThrow();
}); });

11
tsconfig.build.json Normal file
View file

@ -0,0 +1,11 @@
{
"extends": "./tsconfig.json",
"compilerOptions": {
"noEmit": false,
"noEmitOnError": true,
"outDir": "dist/",
"allowImportingTsExtensions": false,
"declaration": true
},
"include": ["./src/**/*"]
}

22
tsconfig.json Normal file
View file

@ -0,0 +1,22 @@
{
"compilerOptions": {
"lib": ["ESNext"],
"target": "ESNext",
"module": "ESNext",
"moduleDetection": "force",
"jsx": "react-jsx",
"allowJs": true,
/* Bundler mode */
"moduleResolution": "bundler",
"allowImportingTsExtensions": true,
"verbatimModuleSyntax": true,
"noEmit": true,
/* Linting */
"skipLibCheck": true,
"strict": true,
"noFallthroughCasesInSwitch": true,
"forceConsistentCasingInFileNames": true
}
}