Skip to content

Commit

Permalink
feat(extension-logging): add http access and method invocation logging
Browse files Browse the repository at this point in the history
- add architecture diagram and examples to README
- improve naming and exporting
- configure the http access logger for REST routes only
  • Loading branch information
raymondfeng committed Jan 15, 2020
1 parent d4a076a commit bab0baf
Show file tree
Hide file tree
Showing 19 changed files with 685 additions and 664 deletions.
56 changes: 27 additions & 29 deletions acceptance/extension-logging-fluentd/package-lock.json

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

Original file line number Diff line number Diff line change
Expand Up @@ -57,8 +57,8 @@ describe('LoggingComponent', () => {
async function givenAppWithCustomConfig() {
app = givenApplication();
app.configure(LoggingBindings.FLUENT_SENDER).to({
host: process.env.FLUENTD_SERVICE_HOST || '127.0.0.1',
port: +(process.env.FLUENTD_SERVICE_PORT_TCP || 0) || 24224,
host: process.env.FLUENTD_SERVICE_HOST ?? '127.0.0.1',
port: +(process.env.FLUENTD_SERVICE_PORT_TCP ?? 24224),
timeout: 3.0,
reconnectInterval: 600000, // 10 minutes
});
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,6 @@ import {GenericContainer, StartedTestContainer} from 'testcontainers';
export const ROOT_DIR = path.join(__dirname, '../../../fixtures');
export const ETC_DIR = path.join(ROOT_DIR, 'etc');

/* eslint-disable require-atomic-updates */
async function startFluentd() {
if (process.env.FLUENTD_SERVICE_HOST != null) return;
const container = await new GenericContainer(
Expand Down
99 changes: 89 additions & 10 deletions extensions/logging/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,10 @@ This module contains a component provides logging facilities based on
> using `0.x.y` versions. Their APIs and functionality may be subject to
> breaking changes in future releases.
## Architecture overview

![logging-component](logging-component.png)

## Installation

```sh
Expand All @@ -34,21 +38,67 @@ In the constructor, add the component to your application:
this.component(LoggingComponent);
```

The component contributes bindings with keys listed below:
Now your application can add a controller as follows to leverage the logging
facilities:

```ts
import {inject} from '@loopback/context';
import {Logger, logInvocation} from '@loopback/extension-logging';
import {get, param} from '@loopback/rest';

class MyController {
// Inject a winston logger
@inject(LoggingBindings.WINSTON_LOGGER)
private logger: Logger;

// http access is logged by a global interceptor
@get('/greet/{name}')
// log the `greet` method invocations
@logInvocation()
greet(@param.path.string('name') name: string) {
return `Hello, ${name}`;
}

@get('/hello/{name}')
hello(@param.path.string('name') name: string) {
// Use the winston logger explicitly
this.logger.log('info', `greeting ${name}`);
return `Hello, ${name}`;
}
}
```

## Configure the logging component

The logging component can be configured as follows:

```ts
app.configure(LoggingBindings.COMPONENT).to({
enableFluent: false, // default to true
enableHttpAccessLog: true, // default to true
});
```

The component contributes bindings with keys declared in `LoggingBindings`
namespace below:

- LoggingBindings.FLUENT_SENDER - A fluent sender
- LoggingBindings.WINSTON_LOGGER - A winston logger
- LoggingBindings.WINSTON_TRANSPORT_FLUENT - A fluent transport for winston
- FLUENT_SENDER - A fluent sender
- WINSTON_LOGGER - A winston logger
- WINSTON_TRANSPORT_FLUENT - A fluent transport for winston
- WINSTON_INTERCEPTOR - A local interceptor set by `@logInvocation` to log
method invocations
- WINSTON_HTTP_ACCESS_LOGGER - A global interceptor that logs http access with
[Morgan](https://github.com/expressjs/morgan) format

The fluent sender and transport for winston can be configured against
`LoggingBindings.FLUENT_SENDER`:
`FLUENT_SENDER`:

```ts
import {LoggingBindings} from '@loopback/extension-logging';

app.configure(LoggingBindings.FLUENT_SENDER).to({
host: process.env.FLUENTD_SERVICE_HOST || 'localhost',
port: +(process.env.FLUENTD_SERVICE_PORT_TCP || 0) || 24224,
host: process.env.FLUENTD_SERVICE_HOST ?? 'localhost',
port: +(process.env.FLUENTD_SERVICE_PORT_TCP ?? 24224),
timeout: 3.0,
reconnectInterval: 600000, // 10 minutes
});
Expand All @@ -75,9 +125,17 @@ points:
```ts
import {extensionFor} from '@loopback/core';
import {format} from 'winston';
import {WINSTON_FORMAT, WINSTON_TRANSPORT} from '@loopback/extension-logging';

const myFormat: Format = ...;
import {
WINSTON_FORMAT,
WINSTON_TRANSPORT,
WinstonFormat,
WinstonTransports,
} from '@loopback/extension-logging';

const myFormat: WinstonFormat = format((info, opts) => {
console.log(info);
return false;
})();

ctx
.bind('logging.winston.formats.myFormat')
Expand All @@ -87,6 +145,27 @@ ctx
.bind('logging.winston.formats.colorize')
.to(format.colorize())
.apply(extensionFor(WINSTON_FORMAT));

const consoleTransport = new WinstonTransports.Console({
level: 'info',
format: format.combine(format.colorize(), format.simple()),
});
ctx
.bind('logging.winston.transports.console')
.to(consoleTransport)
.apply(extensionFor(WINSTON_TRANSPORT));
```

If no transport is contributed, the winston logger uses the
[console](https://github.com/winstonjs/winston/blob/master/docs/transports.md#console-transport).

The access log interceptor can also be configured to customize
[Morgan format and options](https://github.com/expressjs/morgan#morganformat-options):

```ts
ctx
.configure(LoggingBindings.WINSTON_HTTP_ACCESS_LOGGER)
.to({format: 'combined'});
```

## Contributions
Expand Down
Binary file added extensions/logging/logging-component.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading

0 comments on commit bab0baf

Please sign in to comment.