Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/flowcore-io/library-flowcore-microservice-ts
The flowcore microservice library is designed to provide strong opinions on how to handle configuration, logging, health checks, metrics, and observability.
https://github.com/flowcore-io/library-flowcore-microservice-ts
microservice nestjs typescript
Last synced: 2 months ago
JSON representation
The flowcore microservice library is designed to provide strong opinions on how to handle configuration, logging, health checks, metrics, and observability.
- Host: GitHub
- URL: https://github.com/flowcore-io/library-flowcore-microservice-ts
- Owner: flowcore-io
- License: mit
- Created: 2022-12-16T12:45:06.000Z (about 2 years ago)
- Default Branch: main
- Last Pushed: 2024-01-22T15:50:07.000Z (12 months ago)
- Last Synced: 2024-11-01T14:40:45.684Z (2 months ago)
- Topics: microservice, nestjs, typescript
- Language: TypeScript
- Homepage:
- Size: 481 KB
- Stars: 2
- Watchers: 2
- Forks: 0
- Open Issues: 0
-
Metadata Files:
- Readme: README.md
- Changelog: CHANGELOG.md
- License: LICENSE.txt
Awesome Lists containing this project
README
![Build](https://github.com/flowcore-io/library-flowcore-microservice-ts/actions/workflows/publish.yml/badge.svg)
# Flowcore Microservice
The flowcore microservice library is designed to provide strong opinions on how to handle configuration, logging, health
checks, metrics, and
observability. It provides a set of modules that allow you to quickly set up services.The library is particularly well-suited for use in microservice architectures, where it can help you
quickly and easily set up the essential components of configuration, logging, health checks, metrics, and observability.## Installation
install with npm:
```bash
npm install @flowcore/microservice @nestjs/core @nestjs/common @nestjs/terminus
```or yarn:
```bash
yarn add @flowcore/microservice @nestjs/core @nestjs/common @nestjs/terminus
```## Usage
The library provides a set of modules that allow you to quickly set up services. The modules are:
- [Configuration](#configuration)
- [Logging](#logging)
- [Health](#health)
- [Metrics](#metrics)
- [Observability](#observability)### Configuration
The configuration library uses zod to validate the configuration. It provides a set of primitives that can be used to
create configuration schemas. The library also provides a configuration Module that can load and validate configuration
from environment variables.To use the module first create a configuration schema.
```typescript
// config file
import {z} from "zod";
import {ConfigurationSchema} from "@flowcore/microservice";export const SomeConfigurationShape = z.object({
someKey: z.string(),
});
export type SomeConfiguration = z.infer;export class SomeConfigurationSchema extends ConfigurationSchema {
context = "some-context";
linking = {
someKey: {
env: "SOME_KEY",
default: "some-default-value", // optional, only if you want to override the default value when loading the schema
},
};
shape = SomeConfigurationShape;// this is optinal, but can be used to override the defaults specified in the linking
constructor(overrideDefaults?: { [key: string]: any }) {
super();
ConfigurationSchema.override(this.linking, overrideDefaults);
}
}```
> **NB!** If you want to be able to override the default of a configuration value you can use the `ConfigurationSchema`
> class to set the default, not the `zod` schema and include the constructor specified above.then create a module builder that uses the configuration schema.
```typescript
// module builder
import {SomeModule} from "./some.module";
import {BaseBuilder, ConfigService} from "@flowcore/microservice";export class SomeModuleBuilder extends BaseBuilder {
requiredContext = ["some-context"];override build() {
super.build();if (!this.config) {
throw new Error(`Missing config for ${this.constructor.name}`);
}return SomeModule.registerAsync({
imports: [this.config],
inject: [ConfigService],
useFactory: (config: ConfigService) => ({
someKey: config.schema.someKey,
}),
});
}
}
```> **requiredContext** is used when you want the configuration to be required. If the configuration is not provided the
> module will throw an error.Then to use the builder and configuration in your service.
```typescript
// app module
import {Module} from "@nestjs/common";
import {SomeModuleBuilder} from "./some-builder";
import {SomeConfigurationSchema} from "./some-config";
import {
ConfigFactory,
ConfigModule,
} from "@flowcore/microservice";const config = ConfigModule.forRoot(
new ConfigFactory()
// ... other schemas
.withSchema(SomeConfigurationSchema, /* optional override defaults { path-to-key: new default value, } */)
// ... other schemas
);@Module({
imports: [
config,
// ... other modules
new SomeModuleBuilder().withConfig(config).build(),
// ... other modules
],
controllers: [],
providers: [],
})
export class AppModule {
}```
The configuration module has a `DefaultAppConfigurationSchema` that will be applied unless explicitly set to false. The
default configuration schema will load the following environment variables:- `PORT` - the port the service will listen on for http endpoints, for example /health and /metrics (default: 3000)
> this is exported as the `DefaultAppConfiguration` interface in the `@flowcore/microservice` package. And can be used
> when accessing the configuration through `useFactory`.### Logging
The logging library uses `winston` and `nest-winston` to provide logging. It provides configuration and injection
mechanics for the winston logger.To use the module load the configuration schema into the configuration factory and import the logging module using the
builder.```typescript
// app module
import {Module} from "@nestjs/common";
import {
ConfigFactory,
ConfigModule,
LoggerModuleBuilder,
LoggerModuleConfigurationSchema,
} from "@flowcore/microservice";const config = ConfigModule.forRoot(
new ConfigFactory()
// ... other schemas
.withSchema(LoggerModuleConfigurationSchema),
// ... other schemas
);@Module({
imports: [
config,
// ... other modules
new LoggerModuleBuilder().withConfig(config).build(),
// ... other modules
],
controllers: [],
providers: [],
})
export class AppModule {
}```
the logging module will load the following environment variables:
- `LOG_LEVEL` - the log level to use for the logger (default: info)
- `LOG_PRETTY_PRINT` - whether to pretty print the logs (default: false)
- `LOG_USE_LABELS` - whether to use labels in the logs (default: false)> this is exported as the `LoggerModuleConfiguration` interface in the `@flowcore/microservice` package. And can be used
> when accessing the configuration through `useFactory`.### Health
The health module uses `@nestjs/terminus` to provide health checks. It provides decorator that can be used to collect
health checks.to use the module, first create a health controller.
```typescript
// health controller
import {Controller, Get} from "@nestjs/common";
import {HealthCheck, HealthCheckResult} from "@nestjs/terminus";
import {HealthService} from "@flowcore/microservice";@Controller("health")
export class HealthController {
constructor(private health: HealthService) {
}@Get()
@HealthCheck()
async check(): Promise {
return this.health.check();
}
}```
> Remember to add an access decorator to the health check endpoint if you are using authentication in your application.
Then import it and the `HealthModule` into your application module.
```typescript
// app module
import {Module} from "@nestjs/common";
import {
ConfigFactory,
ConfigModule,
HealthModuleBuilder,
} from "@flowcore/microservice";
import {HealthController} from "./health.controller";@Module({
imports: [
new HealthModuleBuilder().usingController(HealthController).build(),
],
controllers: [],
providers: [],
})
export class AppModule {
}
```Then to add health checks to the service, use the `@HealthCheckIndicator` decorator.
```typescript
// some health check
import {HealthCheckIndicator, CheckHealth} from "@flowcore/microservice";
import {Injectable} from "@nestjs/common";
import {HealthIndicator, HealthIndicatorResult} from "@nestjs/terminus";@Injectable()
@HealthCheckIndicator()
class HealthyService extends HealthIndicator implements CheckHealth {
async isHealthy(): Promise {
// do some health check logic
return this.getStatus(/* some key */, /* true or false */);
}
}
```> Remember to add the `HealthModule` to the module imports.
### Metrics
The metrics module is a wrapper around `@willsoto/nestjs-prometheus` to provide metrics for the service. It provides a
module that can be configured and imported into the application module.To use the module, first create a metrics controller.
```typescript
// metrics controller
import {Controller, Get, Res} from "@nestjs/common";
import {PrometheusController} from "@flowcore/microservice";
import {Response} from "express";@Controller()
export class MetricsController extends PrometheusController {
@Get()
async index(@Res() response: Response) {
return super.index(response);
}
}
```Then import it and the `MetricsModule` into your application module using the builder.
```typescript
// app module
import {Module} from "@nestjs/common";
import {MetricsController} from "./metrics.controller";
import {MetricsModuleBuilder} from "@flowcore/microservice";@Module({
imports: [
// ... other modules
new MetricsModuleBuilder().usingController(MetricsController).build()
// ... other modules
],
controllers: [],
providers: [],
})
export class AppModule {
}
```to use the metrics module, follow the instructions in
the [`@willsoto/nestjs-prometheus`](https://www.npmjs.com/package/@willsoto/nestjs-prometheus) package.> The `createCounterProvider` and the `Counter` class are exported from the `@flowcore/microservice` package, same goes
> for `Gauge`,`Histogram` and`Summary`. So you can import them from there instead of the `@willsoto/nestjs-prometheus`
> and `prom-client` packages.To use default node metrics exported from the `@willsoto/nestjs-prometheus` package, use the `withDefaultMetrics` method
when building the metrics module.To add default labels to all metrics, use the `withDefaultLabels` method when building the metrics module.
### Observability (Deprecated)
the observability module has been marked as deprecated, and will be implemented in a new version using Open Telemetry. For now, traces are implemented using eBPF from Groundcover.