0.4.0 • Published 2 days ago

echoed v0.4.0

Weekly downloads
-
License
MIT
Repository
github
Last release
2 days ago

Echoed

Observable Integration Testing using OpenTelemetry on top of Jest/Playwright/Cypress.

Table of Contents

Features

Echoed enhances Integration testing, aka API testing with the following features:

  • Effortless Test Troubleshooting: Quickly identify issues in failed tests by visualizing OpenTelemetry's traces and logs.
  • YAML Supported: Write tests effortlessly using YAML and easily expand functionality through your plugins.
  • Coverage Analysis: Gain insights into the coverage of your API endpoints based on OpenAPI or Protocol Buffers specifications.
  • Detect Propagation Leaks: Uncover spans that don't propagate OpenTelemetry's context to their children.
  • Validate Spans: Validate span's fields, such as SQL or requests going outside.
  • CI-Friendly: Integrates with CI without relying on external services.
  • IDE Debugging: Debug your tests in your preferred IDE, leveraging TypeScript/JavaScript's built-in debugging capabilities.
  • Code Compatibility: No need to modify your existing tests.
  • Parallel Execution: Boost by executing tests in parallel.

How Echoed Works

Echoed starts a local server to gather data through OpenTelemetry when test is started.
Throughout the testing process, Echoed captures OpenTelemetry's traces and logs.
Once tests finish, Echoed generates an HTML report for the test.

How Echoed Works

Screenshots

Echoed generates HTML that visualizes OpenTelemetry traces for each request in tests.
Explore the screenshots below to see how it looks:

  • Trace and logs of the Test
    Screenshot of Echoed's log detail Screenshot of Echoed's log detail
  • Coverage per service
    Screenshot of Echoed's coverage

Installation

Echoed offers several installation methods depending on your needs:

Option 1. Create From Template (Jest)

  1. Initialize a new directory using our template:
    mkdir my_test_directory && cd my_test_directory
    npm create echoed@latest
  2. Review the example tests and run them by following instructions in the generated README.md:
    cat README.md
  3. Run test after compiling YAML tests:
    npm run compile && npm run test
  4. Once you're familiar, remove the example directory and begin crafting your own tests:
    rm -rf ./example

Other Options

Refer to the following documents for other installation methods:

How to Use

Jest

YAML

You can write tests using YAML, and Echoed will convert them into Jest tests.
Compilation flow

The YAML below makes a request to http://localhost:8080/api/cart and validates the response.

variable:
  productId: OLJCESPC7Z
scenarios:
  - name: Get product detail
    steps:
      - description: fetch /products/{id}
        act:
          runner: fetch
          argument:
            endpoint: /products/${productId}
        assert:
          - expect(_.jsonBody.id).toBe(productId)

For more details, refer to the documentation or examples.

Make Tests Observable

You can write Jest tests in TypeScript too.

To generate an HTML report visualizing API traces, no additional code is needed.
Simply write your Jest tests as usual.

describe("Awesome test", () => {
  it("should pass", async () => {
    const response = await fetch(`http://localhost:8080/api/cart`);
    expect(response.status).toBe(200);

    const body = await response.json();
    expect(body.items.length).toBe(0);
  });
});

The code above produces an HTML report illustrating a trace for the requested endpoint (http://localhost:8080/api/cart).

Test OpenTelemetry's Spans

In addition to the HTML output, Echoed offers a method for testing OpenTelemetry spans.
Use the waitForSpan function to obtain a span that matches your needs.

describe("Awesome test", () => {
  it("should create an OpenTelemetry gRPC span", async () => {
    const response = await fetch(`http://localhost:8080/api/products`);
    expect(response.status).toBe(200);

    const span = await waitForSpan(response, {
      name: "oteldemo.ProductCatalogService/ListProducts",
    });
    
    const productsCount = span.attributes.find(attr => attr.key === "app.products.count");
    expect(productsCount?.value?.intValue).toBe(10);
  });
});

The code above waits for a span and compares it using the expect statement

Other Examples

For more examples, refer to documentation.

Playwright

You can write Playwright tests in TypeScript too.

YAML

You can write tests using YAML, and Echoed will convert them into Playwright tests.

Compilation flow

The YAML below opens http://localhost:8080 and validates DOM elements.

scenarios:
  - name: Validate Homepage
    fixtures:
      - page
    steps:
      - description: Check product list is shown
        act:
          raw: await page.goto("http://localhost:8080")
        assert:
          - expectToBeVisible: "[data-cy=home-page]"
          - expectToHaveCount:
              selector: "[data-cy=product-list] [data-cy=product-card]"
              count: 10

For more details, refer to the documentation or examples.

Make Tests Observable

You can write Playwright tests in TypeScript too.

To generate an HTML report visualizing API traces, replace test of Playwright to Echoed's to intercept requests.

// import { test } from "@playwright/test"; <- Replace this line
import { test } from "echoed/playwright/test";

test("opens home page", async ({ page }) => {
  await page.goto("http://localhost:8080/");
  await expect(page).toHaveTitle("OTel demo");

  const productList = page.locator("[data-cy=product-card]");
  await expect(productList).toHaveCount(10);
});

The code above produces an HTML report illustrating traces when opening the home page(http://localhost:8080).

Test OpenTelemetry's Spans

In addition to the HTML output, Echoed offers a method for testing OpenTelemetry spans.
Use the waitForSpanCreatedIn function to obtain a span that matches your needs.

test("creates an OpenTelemetry gRPC span", async ({ page }) => {
  await page.goto("http://localhost:8080/");
  await expect(page).toHaveTitle("OTel demo");

  const span = await waitForSpanCreatedIn(
    page.context(),
    "http://localhost:8080/api/products",
    { name: "oteldemo.ProductCatalogService/ListProducts" },
  );
  
  const rpcSystem = span.attributes.find(
    (attr) => attr.key === "app.products.count",
  );
  expect(rpcSystem?.value?.intValue).toBe(10);
});

The code above waits for a span that links to the request to http://localhost:8080/api/products and compares it using the expect statement

Other Examples

For more examples, refer to documentation.

Cypress

You can write Cypress tests in TypeScript too.

Make Tests Observable

To generate an HTML report visualizing API traces, no additional code is needed.
Simply write your Cypress tests as usual.

it("opens home page", () => {
  cy.visit("http://localhost:8080");
  cy.title().should("eq", "OTel demo");
  
  cy.get("[data-cy=product-card]").should("have.length", 10);
});

The code above produces an HTML report illustrating traces when opening the home page(http://localhost:8080).

Test OpenTelemetry's Spans

In addition to the HTML output, Echoed offers a method for testing OpenTelemetry spans.
Use the waitForSpan command to obtain a span that matches your needs.

it("creates an OpenTelemetry gRPC span", () => {
  cy.visit("http://localhost:8080");
  cy.title().should("eq", "OTel demo");

  cy.waitForSpan(
    "http://localhost:8080/api/products",
    { name: "oteldemo.ProductCatalogService/ListProducts" },
  ).then((span) => {
    const rpcSystem = span.attributes.find(
      (attr) => attr.key === "app.products.count",
    );
    expect(rpcSystem?.value?.intValue).to.eq(10);
  });
});

The code above waits for a span that links to the request to http://localhost:8080/api/products and compares it using the expect statement

Other Examples

For more examples, refer to documentation.

Analyze Coverage

You can get coverage of your HTTP and gRPC endpoints based on OpenAPI or Protocol Buffers specifications.
By configuring the openapi or proto option in your .echoed.yml file, Echoed analyzes the coverage of your tests and generates a report.
For more option, refer to the Configuration section.

services:
  - name: frontend
    namespace: opentelemetry-demo
    openapi: "./example/opentelemetry-demo/src/frontend/schema.yaml"
  - name: cartservice
    namespace: opentelemetry-demo
    proto:
      filePath: "./example/opentelemetry-demo/pb/demo.proto"
      services:
        - oteldemo.CartService

Using Echoed without OpenTelemetry

While Echoed's primary feature is to troubleshoot or analyze tests by visualizing OpenTelemetry data, it can also be used to write tests in YAML.
To add YAML tests into existing tests, simply create a .echoed.yml file for configuration and run npx echoed compile.

Alternatively, if you wish to create example tests without OpenTelemetry, you can do so using the following commands:

# Create example tests
npm create echoed@latest -- --template jest-no-otel

# Compile YAML to TypeScript and run tests
npx echoed compile
npx jest

Or for Playwright:

# Create example tests
npm create echoed@latest -- --template playwright-no-otel

# Compile YAML to TypeScript and run tests
npx echoed compile
npx playwright test

Configuration

Echoed can be configured at .echoed.yml in the root of your project.
Explore available options here.