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 🙏

© 2026 – Pkg Stats / Ryan Hefner

@saptools/cf-logs

v0.1.3

Published

Fetch, stream, parse, and cache SAP BTP Cloud Foundry application logs

Readme

📜 @saptools/cf-logs

Turn SAP BTP Cloud Foundry logs into a reusable engine, not a one-off UI feature.

Fetch snapshots, stream live output, parse plain-text and JSON logs, normalize router access rows, redact credentials, and persist bounded log state to disk through one CLI and one typed Node.js API.

npm version license node install size types

InstallQuick StartCLIAPIStoreSecurity


✨ Features

  • 📥 Recent snapshots — run cf logs --recent, redact sensitive values, parse the result, and optionally persist it
  • 📡 Live streams — wrap cf logs <app> with batching, reconnection, bounded in-memory state, and typed events
  • 🧠 Real log parsing — handle plain text, JSON logs, multiline continuations, and router access metadata such as method, request, status, latency, tenant, client IP, and request ID
  • 🗃️ Bounded local store — write redacted snapshots to ~/.saptools/cf-logs-store.json with atomic file updates and locking
  • 🧩 CLI and typed API — use the package from shell scripts, VSCode extensions, Node services, or test runners
  • 🧪 Fake-backed E2E coverage — snapshot, parse, and stream flows are verified without live SAP access

📦 Install

# Global CLI
npm install -g @saptools/cf-logs

# Or as a dependency
npm install @saptools/cf-logs
# pnpm add @saptools/cf-logs
# yarn add @saptools/cf-logs

[!NOTE] Requires Node.js >= 20 and the official cf CLI on PATH.


🚀 Quick Start

# 1. Export credentials used for cf api/auth
export SAP_EMAIL="[email protected]"
export SAP_PASSWORD="sample-password"

# 2. Fetch a structured snapshot
cf-logs snapshot \
  --region ap10 \
  --org sample-org \
  --space sample \
  --app demo-app \
  --json

# 3. Start a live stream as line-delimited JSON events
cf-logs stream \
  --region ap10 \
  --org sample-org \
  --space sample \
  --app demo-app \
  --json

If you already know the CF API endpoint, replace --region ap10 with --api-endpoint https://api.cf.ap10.hana.ondemand.com.


🧰 CLI

Shared targeting flags

Most commands use the same target shape:

| Flag | Description | | --- | --- | | --region <key> | CF region key such as ap10 | | --api-endpoint <url> | Explicit CF API endpoint instead of a region key | | --org <name> | CF org name | | --space <name> | CF space name | | --app <name> | CF app name | | --email <value> | Override SAP_EMAIL | | --password <value> | Override SAP_PASSWORD |

--region or --api-endpoint is required. Credentials default to SAP_EMAIL and SAP_PASSWORD.

[!WARNING] Prefer the environment variables over --password <value>. Process arguments are visible to other users on the machine via tools like ps, so passing credentials inline can leak them. The --password flag is kept as an explicit override for one-off scripts only.

cf-logs snapshot

Fetch recent logs for one app. By default the command prints bounded redacted raw text. Use --json for structured rows and --save to persist the snapshot.

cf-logs snapshot \
  --region ap10 \
  --org sample-org \
  --space sample \
  --app demo-app \
  --json \
  --save

| Flag | Description | | --- | --- | | --json | Emit a full JSON snapshot object | | --save | Persist the redacted snapshot to the local store | | --log-limit <count> | Maximum parsed rows and bounded raw-text budget |

cf-logs stream

Start a live log stream for one app. In JSON mode the command emits line-delimited events:

  • {"type":"state",...}
  • {"type":"lines",...}
cf-logs stream \
  --region ap10 \
  --org sample-org \
  --space sample \
  --app demo-app \
  --json

Useful stream options:

| Flag | Description | | --- | --- | | --json | Emit line-delimited JSON events | | --save | Persist bounded redacted stream appends to the local store | | --max-lines <count> | Stop after emitting N streamed lines | | --log-limit <count> | Maximum parsed rows and bounded raw-text budget | | --flush-interval-ms <ms> | Batch window before append events are emitted | | --retry-initial-ms <ms> | Initial reconnect delay after unexpected stream exits | | --retry-max-ms <ms> | Maximum reconnect delay |

cf-logs parse

Parse a local file or stdin into structured rows.

# Parse a file
cf-logs parse --input ./sample.log

# Parse stdin
cat ./sample.log | cf-logs parse

| Flag | Description | | --- | --- | | --input <path> | Read from a file instead of stdin | | --raw | Print bounded raw input instead of structured rows | | --log-limit <count> | Maximum parsed rows and bounded raw-text budget |

cf-logs apps

List started apps with running instances for one org/space.

cf-logs apps --region ap10 --org sample-org --space sample --json

cf-logs store path

Print the package-managed log store path.

cf-logs store path

cf-logs store list

Inspect cached store entries.

cf-logs store list
cf-logs store list --json

cf-logs store clear

Remove every cached entry from the package-managed log store. The store file remains in place but is rewritten with an empty entries list.

cf-logs store clear

cf-logs --version

Print the installed @saptools/cf-logs semantic version.

cf-logs --version

🧑‍💻 Programmatic Usage

Parse logs directly

import { filterRows, parseRecentLogs } from "@saptools/cf-logs";

const rows = parseRecentLogs(`
2026-04-12T09:14:40.00+0700 [APP/PROC/WEB/0] OUT ready
2026-04-12T09:14:41.00+0700 [APP/PROC/WEB/0] OUT {"level":"error","logger":"samplelogger","timestamp":"2026-04-12T02:14:41.000Z","msg":"save failed","type":"log"}
`);

const errors = filterRows(rows, { level: "error" });
console.log(errors[0]?.message);

Drive snapshots and streams from code

import { CfLogsRuntime } from "@saptools/cf-logs";

const runtime = new CfLogsRuntime({
  persistSnapshots: true,
  persistStreamAppends: true,
  retryInitialMs: 1_000,
  retryMaxMs: 20_000,
});

runtime.setSession({
  region: "ap10",
  email: process.env["SAP_EMAIL"] ?? "",
  password: process.env["SAP_PASSWORD"] ?? "",
  org: "sample-org",
  space: "sample",
});

runtime.setAvailableApps([{ name: "demo-app", runningInstances: 1 }]);

runtime.subscribe((event) => {
  if (event.type === "append") {
    process.stdout.write(`${event.lines.join("\n")}\n`);
  }
});

const snapshot = await runtime.fetchSnapshot("demo-app");
console.log(snapshot.rows.length);

await runtime.setActiveApps(["demo-app"]);

Lower-level helpers

import {
  fetchRecentLogs,
  fetchStartedAppsViaCfCli,
  readStore,
  resolveApiEndpoint,
} from "@saptools/cf-logs";

const apiEndpoint = resolveApiEndpoint({ region: "ap10" });
const apps = await fetchStartedAppsViaCfCli({
  apiEndpoint,
  email: process.env["SAP_EMAIL"] ?? "",
  password: process.env["SAP_PASSWORD"] ?? "",
  org: "sample-org",
  space: "sample",
});

const rawLogs = await fetchRecentLogs({
  apiEndpoint,
  email: process.env["SAP_EMAIL"] ?? "",
  password: process.env["SAP_PASSWORD"] ?? "",
  org: "sample-org",
  space: "sample",
  app: apps[0]?.name ?? "demo-app",
});

const store = await readStore();
console.log(rawLogs.length, store.entries.length);

📁 Store File

The package-managed store lives here:

~/.saptools/cf-logs-store.json

It contains redacted, bounded entries keyed by:

  • apiEndpoint
  • org
  • space
  • app

Each entry stores:

  • rawText
  • fetchedAt
  • updatedAt
  • rowCount
  • truncated

The store is an implementation detail. Prefer readStore(), persistSnapshot(), cf-logs store path, or cf-logs store list over parsing the file directly.


🔐 Security Notes

  • The CLI and runtime redact the current SAP email and password before emitting or persisting logs.
  • Persisted snapshots are bounded and written with file locking plus atomic replace semantics.
  • The store file is not safe for public repositories. Even after redaction, it still reveals app names, org names, spaces, endpoints, and log content.
  • If your application logs contain additional secrets beyond SAP credentials, add custom runtime redaction rules before persisting or forwarding log output.

❓ FAQ

No. The package is intentionally UI-agnostic. It exposes a CLI, a parser, a runtime engine, and store helpers that can be used by VSCode, terminal tools, tests, or backend processes.

Because parsing, streaming, redaction, and store management are reusable engine concerns. Keeping them in a package makes the IDE layer smaller, easier to test, and easier to evolve.

No. Persistence is opt-in. Use --save in the CLI or enable persistSnapshots / persistStreamAppends in CfLogsRuntime.

Yes. Pass --api-endpoint <url> in the CLI or apiEndpoint in the Node.js API.


🛠️ Development

From the monorepo root:

pnpm install
pnpm --filter @saptools/cf-logs cspell
pnpm --filter @saptools/cf-logs lint
pnpm --filter @saptools/cf-logs typecheck
pnpm --filter @saptools/cf-logs test:unit
pnpm --filter @saptools/cf-logs test:e2e
pnpm --filter @saptools/cf-logs build
pnpm --filter @saptools/cf-logs check

🤝 Author

Maintained by dongtran.