Skip to content

Commit ef4457f

Browse files
committed
initial commit
0 parents  commit ef4457f

14 files changed

+295
-0
lines changed

.github/workflows/ci.yml

+37
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,37 @@
1+
name: CI
2+
on:
3+
push:
4+
branches:
5+
- master
6+
jobs:
7+
test:
8+
runs-on: ubuntu-latest
9+
steps:
10+
- name: Checkout 🛎️
11+
uses: actions/[email protected]
12+
with:
13+
persist-credentials: false
14+
- uses: actions/setup-node@v2
15+
with:
16+
node-version: "18.x"
17+
cache: "npm"
18+
19+
- name: Install
20+
run: env NODE_ENV=development npm install --prefer-offline --no-audit
21+
- name: Test
22+
run: npm run typecheck && npm run test
23+
24+
- name: Build
25+
run: env NODE_ENV=production npm run build
26+
- run: jq -s 'add' package.json package-publish.json > package-final.json && rm -rf package.json && mv package-final.json package.json
27+
28+
- id: Publish
29+
uses: JS-DevTools/npm-publish@v1
30+
with:
31+
registry: "https://registry.npmjs.org/"
32+
token: ${{ secrets.NPM_TOKEN }}
33+
access: "public"
34+
check-version: true
35+
- if: steps.publish.outputs.type != 'none'
36+
run: |
37+
echo "Version changed: ${{ steps.publish.outputs.old-version }} => ${{ steps.publish.outputs.version }}"

.gitignore

+4
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,4 @@
1+
node_modules
2+
.vscode
3+
.DS_Store
4+
dist

Makefile

+7
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
2+
.SHELL := /usr/bin/bash
3+
4+
.PHONY: build
5+
build:
6+
npx tsc --resolveJsonModule -p ./tsconfig.json --outDir ./dist --emitDeclarationOnly --declaration
7+
node esbuild.mjs

README.md

+14
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,14 @@
1+
# exami
2+
3+
⚡ Ultra lean test framework for browser and node.
4+
5+
[![Version](https://img.shields.io/npm/v/exami.svg?color=success&style=flat-square)](https://www.npmjs.com/package/exami)
6+
[![License: MIT](https://img.shields.io/badge/License-MIT-brightgreen.svg)](https://opensource.org/licenses/MIT)
7+
[![Build Status](https://github.com/abhishiv/exami/actions/workflows/ci.yml/badge.svg)](https://github.com/abhishiv/exami/actions/workflows/ci.yml)
8+
![Badge size](https://deno.bundlejs.com/badge?q=exami&treeshake=[*]&config={%22cdn%22:%22https://cdn.jsdelivr.net/npm%22})
9+
10+
**npm**: `npm i exami`
11+
12+
**cdn**: https://cdn.jsdelivr.net/npm/exami/+esm
13+
14+
---

esbuild.mjs

+22
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,22 @@
1+
import * as esbuild from "esbuild";
2+
3+
(async () => {
4+
await esbuild.build({
5+
entryPoints: ["src/index.ts"], // Specify your entry point(s) here
6+
outfile: "dist/index.js", // Specify the output file here
7+
bundle: true,
8+
packages: "external",
9+
target: "es2022",
10+
format: "esm",
11+
sourcemap: false,
12+
minify: true,
13+
plugins: [
14+
// sassPlugin({
15+
// type: "style",
16+
// transform: postcssModules({
17+
// // ...put here the options for postcss-modules: https://github.com/madyankin/postcss-modules
18+
// }),
19+
// }),
20+
],
21+
});
22+
})();

package-lock.json

+7
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

package-publish.json

+11
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,11 @@
1+
{
2+
"main": "./dist/index.js",
3+
"module": "./dist/index.js",
4+
"types": "./dist/index.d.ts",
5+
"exports": {
6+
".": {
7+
"types": "./dist/index.d.ts",
8+
"import": "./dist/index.js"
9+
}
10+
}
11+
}

package.json

+38
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,38 @@
1+
{
2+
"name": "exami",
3+
"version": "0.8.51",
4+
"author": "Abhishiv Saxena<[email protected]>",
5+
"license": "MIT",
6+
"description": "Ultralean Testing framework for browser and node",
7+
"keywords": [],
8+
"repository": {
9+
"type": "git",
10+
"url": "https://github.com/abhishiv/attorney"
11+
},
12+
"files": [
13+
"./dist"
14+
],
15+
"devDependencies": {
16+
"esbuild": "^0.21.5",
17+
"typescript": "^5.2.2",
18+
"vitest": "^2.0.5"
19+
},
20+
"scripts": {
21+
"build": "make build",
22+
"test": "npx vitest --run --passWithNoTests",
23+
"coverage": "vitest run --coverage --run --passWithNoTests",
24+
"typecheck": "npx tsc --noEmit"
25+
},
26+
"sideEffects": false,
27+
"type": "module",
28+
"main": "./src/index.ts",
29+
"exports": {
30+
".": {
31+
"import": "./src/index.ts"
32+
}
33+
},
34+
"dependencies": {
35+
"buckwheat": "^1.1.2"
36+
},
37+
"peerDependencies": {}
38+
}

src/example.ts

+16
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,16 @@
1+
import { describe, runSuite, printTap } from "./index";
2+
3+
const suite = describe("My first test suite", ({ test }) => {
4+
test("my test1", ({ expect }) => {
5+
expect(4).toBe(4);
6+
});
7+
8+
test("my test2", ({ expect }) => {
9+
expect(4).toBe(4);
10+
});
11+
});
12+
13+
runSuite(suite).then((results) => {
14+
// console.log(results);
15+
console.log(printTap(results));
16+
});

src/index.ts

+51
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,51 @@
1+
import {
2+
Suite,
3+
SuiteMeta,
4+
SuiteHandler,
5+
SuiteAttrs,
6+
TestMeta,
7+
TestHandler,
8+
Results,
9+
} from "./types";
10+
import { expect } from "buckwheat";
11+
12+
export * from "./types/index";
13+
export * from "./tap";
14+
15+
export const describe = (meta: string | SuiteMeta, fn: SuiteHandler) => {
16+
const suite: Suite = {
17+
type: "Suite",
18+
meta: typeof meta === "string" ? { name: meta } : meta,
19+
tests: [],
20+
};
21+
const attrs: SuiteAttrs = {
22+
test: (name: string | TestMeta, fn: TestHandler) => {
23+
suite.tests.push({
24+
type: "Test",
25+
meta: typeof name === "string" ? { name } : name,
26+
fn: fn,
27+
});
28+
},
29+
};
30+
fn(attrs);
31+
return suite;
32+
};
33+
34+
export async function runSuite(suite: Suite) {
35+
const results: Results = [];
36+
suite.tests.forEach(async (test, index) => {
37+
try {
38+
const fn = test.fn({ expect });
39+
const result = await Promise.resolve(fn);
40+
results.push({ index, test, result, type: "passed" });
41+
} catch (e) {
42+
results.push({
43+
index,
44+
test,
45+
type: "failed",
46+
error: e as any,
47+
});
48+
}
49+
});
50+
return results;
51+
}

src/tap.ts

+31
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,31 @@
1+
import { Results } from "./types/index";
2+
3+
export function printTap(arg: Results) {
4+
const results = arg;
5+
const tapOutput: string[] = [];
6+
7+
// Pushing the test plan (total number of tests) into the array
8+
tapOutput.push(`1..${results.length}`);
9+
10+
// Iterating through each test result and pushing TAP lines into the array
11+
results.forEach((result, i) => {
12+
if (result.type === "passed") {
13+
tapOutput.push(`ok ${result.index} - ${result.test.meta.name}`);
14+
} else {
15+
tapOutput.push(`not ok ${result.index} - ${result.test.meta.name}`);
16+
tapOutput.push(` ---`);
17+
tapOutput.push(` name: ${result.error.name}`);
18+
tapOutput.push(` operator: error`);
19+
if (result.error instanceof Error) {
20+
tapOutput.push(` message: ${result.error.message}`);
21+
tapOutput.push(` stack: ${result.error.stack}`);
22+
} else {
23+
tapOutput.push(` actual: ${result.error.actual}`);
24+
tapOutput.push(` expected: ${result.error.expected}`);
25+
}
26+
tapOutput.push(` ...`);
27+
}
28+
});
29+
30+
return tapOutput.join("\n");
31+
}

src/types/index.ts

+29
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,29 @@
1+
import { expect } from "buckwheat";
2+
3+
export type SuiteAttrs = { test: TestDefinition };
4+
export type SuiteHandler = (attrs: SuiteAttrs) => void;
5+
6+
export type TestAttrs = { expect: typeof expect };
7+
export type TestHandler = (attrs: TestAttrs) => void;
8+
9+
export type SuiteMeta = { name: string };
10+
export type TestMeta = { name: string };
11+
12+
export type TestDefinition = (name: string | TestMeta, fn: TestHandler) => void;
13+
14+
export type Test = { type: "Test"; meta: TestMeta; fn: TestHandler };
15+
export type Suite = { type: "Suite"; meta: SuiteMeta; tests: Test[] };
16+
17+
export type PassedResult = {
18+
index: number;
19+
type: "passed";
20+
test: Test;
21+
result: any;
22+
};
23+
export type FailedResult = {
24+
index: number;
25+
type: "failed";
26+
test: Test;
27+
error: { name: string; actual: any; expected: any } | Error;
28+
};
29+
export type Results = (PassedResult | FailedResult)[];

tsconfig.json

+21
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
{
2+
"ts-node": {
3+
"compilerOptions": {
4+
"module": "commonjs"
5+
}
6+
},
7+
"compilerOptions": {
8+
"target": "ESNext",
9+
"useDefineForClassFields": true,
10+
"module": "ESNext",
11+
"moduleResolution": "Node",
12+
"strict": true,
13+
"jsx": "react",
14+
"resolveJsonModule": true,
15+
"isolatedModules": true,
16+
"esModuleInterop": true,
17+
"lib": ["ESNext", "DOM"],
18+
"skipLibCheck": true
19+
},
20+
"include": ["src/**/*.ts", "src/**/*.d.ts", "src/**/*.tsx", "src/**/*.vue"]
21+
}

vite.config.js

+7
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
import { defineConfig } from "vite";
2+
3+
export default defineConfig({
4+
build: {
5+
target: "esnext",
6+
},
7+
});

0 commit comments

Comments
 (0)