npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2025 – Pkg Stats / Ryan Hefner

@aller/pino-gcp-transport

v0.0.3

Published

Pino google cloud structured logging stdout transport

Downloads

184

Readme

pino-gcp-transport

Convert stdout logging to structured json suitable for google cloud logging.

Build

Api

Exported as ESM and commonjs.

compose([options[, Transformation = StructuredTransformation]])

Compose transport to get structured log. Default and named export.

Arguments:

  • options: optional structured transport options
    • ignoreKeys: optional list of pino ignore keys, filters what log line keys are sent to jsonPayload, defaults to ['hostname', 'pid', 'level', 'time', 'msg']
    • destination: optional integer or string, integer 1 = stdout which is default, if a string is used the destination will be considered a log file, e.g. ./logs/test.log
    • append: optional boolean, set to false if you want file test logging to be reset between test runs, passed to sonic-boom
    • [key]: any other options that can be passed to sonic-boom
  • Transformation: optional transformation stream type, defaults to builtin StructuredTransformation stream type, will be called with new. Can be used to override with extended StructuredTransformation type
import compose, { StructuredTransformation } from '@aller/pino-gcp-transport';

getLogTrace(projectId)

Used in mixin to get trace context logging params. Named export.

Arguments:

Returns

Object with structured log keys logging.googleapis.com/trace and logging.googleapis.com/spanId.

import { getLogTrace } from '@aller/pino-gcp-transport';
import pino from 'pino';

export const logger = pino({
  mixin() {
    return { ...getLogTrace(process.env.MYAPP_projectId) };
  },
});

getTraceHeaders([flags = 0])

Used to forward tracing to donwstream calls. Named export.

Arguments:

  • flags: optional positive integer, below 256, tracing flag, 0 = not sampled, 1 = sampled

Returns

Map with trace headers traceparent and legacy x-cloud-trace-context.

import compose, { getTraceHeaders, getLogTrace } from '@aller/pino-gcp-transport';
import pino from 'pino';

export const logger = pino(
  {
    mixin() {
      return { ...getLogTrace(process.env.MYAPP_projectId) };
    },
  },
  compose()
);

await fetch('http://localhost:11434/get', {
  headers: Object.fromEntries(getTraceHeaders()),
}).catch((err) => {
  logger.error(err);
});

getTraceHeadersAsObject([flags = 0])

Same as getTraceHeaders but returns object.

Setup example

import pino from 'pino';
import express from 'express';
import request from 'supertest';

import compose, { middleware, getLogTrace, getTraceHeadersAsObject } from '@aller/pino-gcp-transport';

const logger = pino(
  {
    level: 'trace',
    mixin() {
      return { ...getLogTrace('aller-project-1') };
    },
  },
  compose()
);

const app = express();

app.use(middleware());

app.get('/downstream', async (req, res, next) => {
  try {
    logger.debug('foo');
    // pass your tracing headers to a downstream service
    await request('https://example.com')
      .get('/')
      .set(getTraceHeadersAsObject(req.query.flags ? Number(req.query.flags) : undefined));
    res.send({});
  } catch (err) {
    next(err);
  }
});

app.use('/log/request', (req, res) => {
  logger.info(req);
  res.send({});
});

app.get('/log/error', (req, res) => {
  logger.error(new Error(req.query.message ?? 'expected'));
  res.send({});
});

app.use((err, _req, res, next) => {
  if (!(err instanceof Error)) return next();

  logger.error(err);
  res.status(500).send({ message: err.message });
});

// For testing
await request(app).get('/log/request').expect(200);

Logger example

import config from 'exp-config';
import { pino } from 'pino';

import { getLogTrace } from '@aller/pino-gcp-transport';

const destination = config.logging?.target === 'file' ? `./logs/${config.envName}.log` : 1;

const targets = [];
if (config.logging.prettify) {
  targets.push({
    target: 'pino-pretty',
    options: {
      colorize: true,
      ignore: 'pid,hostname',
      translateTime: "yyyy-mm-dd'T'HH:MM:ss.l",
      destination,
      append: false,
    },
  });
} else {
  targets.push({
    target: '@aller/pino-gcp-transport',
    options: {
      destination,
      append: false,
    },
  });
}

const transport = pino.transport({ targets });

export default pino(
  {
    level: config.logging?.level || 'debug',
    mixin() {
      return { ...getLogTrace(config.projectId) };
    },
  },
  transport
);

The projectId

To be able to log trace the google project id needs to be passed to the mixin function getLogTrace(projectId). Undefined is returned if called without project id.

The project id is used to format the log parameter logging.googleapis.com/trace formatted as projects/${projectId.trim()}/traces/${traceId}

If deploying with terraform you can pick the id from your project resource, e.g:

locals {
  env_pfx = "MYAPP_"
}

resource "google_cloud_run_v2_service" "myapp" {
  template {
    containers {
      image = "gcr.io/cloudrun/hello"

      ports {
        container_port = 3000
      }

      env {
        name  = "NODE_ENV"
        value = var.environment
      }
      env {
        name  = "ENV_PREFIX"
        value = local.env_pfx
      }
      env {
        name  = "INTERPRET_CHAR_AS_DOT"
        value = "_"
      }

      # The google project eventually picked up by exp-config as config.projectId, or just use as process.env.MYAPP_projectId
      env {
        name  = "${local.env_pfx}projectId"
        value = google_project.default.project_id
      }