An open API service indexing awesome lists of open source software.

https://github.com/a11ywatch/sidecar

A11yWatch lite javascript sidecar and client
https://github.com/a11ywatch/sidecar

a11ywatch sidecar

Last synced: 6 months ago
JSON representation

A11yWatch lite javascript sidecar and client

Awesome Lists containing this project

README

          

# sidecar

A11yWatch Lite native javascript [sidecar](https://learn.microsoft.com/en-us/azure/architecture/patterns/sidecar)

## Getting started

Make sure to have either [nodejs](https://nodejs.org/en/) or [bun](https://bun.sh/) installed.
[Protoc](https://grpc.io/docs/protoc-installation/#install-pre-compiled-binaries-any-os) is required to install the crawler, depending on the
OS an automatic installation is performed.

### nodejs

node - v14.0 and up, pref node ^v18.

1. `yarn add @a11ywatch/a11ywatch`.
1. `npx playwright install`

### bun

1. `npx playwright install` # we need pre-install scripts
1. `bun install @a11ywatch/a11ywatch`.

In order to build `@a11ywatch/crawler` >= 0.5.0, you need the `protoc` Protocol Buffers compiler, along with Protocol Buffers resource files.

#### Ubuntu

```bash
sudo apt update && sudo apt upgrade -y
sudo apt install -y protobuf-compiler libprotobuf-dev
```

#### Alpine Linux

```sh
sudo apk add protoc protobuf-dev
```

#### macOS

Assuming [Homebrew](https://brew.sh/) is already installed. (If not, see instructions for installing Homebrew on [the Homebrew website](https://brew.sh/).)

```zsh
brew install protobuf
```

#### optional

Use the packages exported methods, use a [client](https://github.com/a11ywatch/a11ywatch/tree/main/clients), or your own handling of request like curl/fetch.

1. optional: start `mongodb` on port 27017.
1. optional: start `redis` on 6379.
1. optional: add `scripts` to .gitignore (auto generated javascript page fixes) or `A11YWATCH_NO_STORE=false` to disable storage.
1. optional: add `DISABLE_HTTP=true` env variable to prevent http server startup (disables OpenAPI & GraphQL)

## Usage

```ts
import { scan, multiPageScan, crawlList } from "@a11ywatch/a11ywatch";

// single page website scan.
await scan({ url: "https://a11ywatch.com" });

// all pages
await multiPageScan({ url: "https://a11ywatch.com" });

// all pages and subdomains
await multiPageScan({
url: "https://a11ywatch.com",
subdomains: true,
});

// all pages and tld extensions
await multiPageScan({ url: "https://a11ywatch.com", tld: true });

// all pages, subdomains, and tld extensions
await multiPageScan({
url: "https://a11ywatch.com",
subdomains: true,
tld: true,
});

// multi page scan with callback on each result asynchronously
const callback = ({ data }) => {
console.log(data);
};
await multiPageScan(
{
url: "https://a11ywatch.com",
},
callback
);

// crawl a list of static urls - does not perform site wide scans
const pages = await crawlList({
pages: [
"https://a11ywatch.com",
"https://newegg.com",
"https://bun.sh",
"https://nodejs.org",
],
userId: 0,
});
```

You can also drill in to the specific modules like `@a11ywatch/pagemind` (accessibility service) and etc.

```ts
import { detectImage } from "@a11ywatch/mav/ai/detectImage";

// prediect an image from a base64. You can also pass in a url as the url key.
await detectImage({
imageBase64:
"",
height: 50,
width: 50,
});
```

## Jest

Here is an example of using A11yWatch with [jest](https://jestjs.io/).

```ts
describe("accessibility suite", () => {
// can scan a single website url
test("passes web accessibility test for home page", async () => {
const { scan } = await import("@a11ywatch/a11ywatch");

const results = await scan({ url: "https://a11ywatch.com" });

// expect to have no issues
expect(results.data.issues.length).toBe(0);
});

// can scan an entire website
test("passes web accessibility test entire website", async () => {
const { multiPageScan } = await import("@a11ywatch/a11ywatch");

await multiPageScan(
{
url: "https://jeffmendez.com",
userId: data.id,
},
({ data }) => {
const issuesCount = data.issues.filter(
(issue) => issue.type === "error"
).length;
console.info(`${data.url}: ${data.issues.length}`);
expect(issuesCount).toBeLessThan(4);
}
);
});
});
```

## Docker

You can run the suite with docker by running:

```
docker run -p 3280:3280 \
-v ${PWD}:/a11ywatch/conf \
-e SUPER_MODE=true \
a11ywatch/a11ywatch
```

Example docker compose configuration:

```yml
version: "3.9"
services:
a11ywatch:
image: a11ywatch/a11ywatch
ports:
- 3280:3280
```

Validate scan:

```sh
curl --location --request POST 'http://localhost:3280/api/crawl' \
--header 'Transfer-Encoding: chunked' \
--header 'Content-Type: application/json' \
--header 'Authorization: $A11YWATCH_TOKEN' \
--data-raw '{
"websiteUrl": "https://a11ywatch.com",
"subdomains": false,
"tld": false,
"robots": false
}'
```

## Packages exposed

The following packages can be imported to use directly to extend [A11yWatch Lite](https://github.com/a11ywatch/a11ywatch).

1. [`@a11ywatch/core`](https://github.com/a11ywatch/a11ywatch-core)
1. [`@a11ywatch/mav`](https://github.com/a11ywatch/mav)
1. [`@a11ywatch/pagemind`](https://github.com/a11ywatch/pagemind)
1. [`@a11ywatch/crawler`](https://github.com/a11ywatch/crawler)
1. [`@a11ywatch/elastic-cdn`](https://github.com/a11ywatch/elastic-cdn)
1. [`@a11ywatch/client`](https://gitlab.com/j-mendez/a11ywatch-clients/-/tree/main/typescript_api_client)

## Examples

View the [bun example](./examples/bun/) or [nodejs example](./examples/nodejs/) for a getting started point.

## Environmental variables

You can update the env variables below to adjust the gRPC server ports and other application features.

```
# grpc configs
GRPC_HOST_PAGEMIND=127.0.0.1:50052
GRPC_HOST_MAV=127.0.0.1:50053
GRPC_HOST_CRAWLER=127.0.0.1:50055
# store js scripts for CDN
SCRIPTS_ENABLED=false
# disabling AI may drastically improve performance across certain pages
AI_DISABLED=false
# disable core http server API
DISABLE_HTTP=true
# unlock all features
SUPER_MODE=true
# prevent auto starting suite - must use initApplication manually when ready
A11YWATCH_AUTO_START=true
# disable storing scripts
A11YWATCH_NO_STORE=true
# disable db install if already started locally
DISABLE_POSTINSTALL=true
# do not store data and use memory db
A11YWATCH_MEMORY_ONLY=true
# enable screen recording to a directory
PAGEMIND_RECORD=true
# set the chrome host
CHROME_HOST=
```

## Pipelines

We use [dagger](https://docs.dagger.io/) to build pipelines for builds, test, and deploys.

```
# deps check
dagger do deps
# build img
dagger do build
# push img locally
# make sure to have localhost listening - `docker run -d -p 2222:5000 --restart=always --name localregistry registry:2`
dagger do push
# now you can run `docker run localhost:2222/a11ywatch:latest`
```

## Client

We bundled the client for a one stop install for javascript at `@a11ywatch/a11ywatch/client`.

```ts
import { setAPIToken, setA11yWatchURL, crawlWebsite, scanWebsite } from "@a11ywatch/a11ywatch/client"

// set at runtime to a new token - if you use the env variable below it is also set without the method
setAPIToken(process.env.A11YWATCH_TOKEN)
setA11yWatchURL("https://api.a11ywatch.com")

// single page audit
const audit = await scanWebsite({ url: "https://mywebsite.com" })
console.log(audit)

// stream multi page full website audit results
await crawlWebsite({ url: "https://mywebsite.com" }, (audit) => {
console.log(audit)
})
```

## Help

When on node v18 and above the flag `--no-experimental-fetch` is required ex: `node --no-experimental-fetch server.js`.

If you run into issues with modules not loading try deleting all of the `node_modules` and running `npm install` again.
Upon switching node versions certain modules need to be re-installed for the native compilation to work.

The project starts a valid [A11yWatch Lite](https://github.com/a11ywatch/a11ywatch) instance.

## LICENSE

check the license file in the root of the project.