Skip to content

Commit

Permalink
feat: adding skeleton and report event as example
Browse files Browse the repository at this point in the history
  • Loading branch information
barbmarcio committed Jul 5, 2024
1 parent d5dde6b commit 58e9e45
Show file tree
Hide file tree
Showing 12 changed files with 491 additions and 3 deletions.
172 changes: 172 additions & 0 deletions .gitignore
Original file line number Diff line number Diff line change
@@ -0,0 +1,172 @@
# Based on https://raw.githubusercontent.com/github/gitignore/main/Node.gitignore

# Logs

logs
_.log
npm-debug.log_
yarn-debug.log*
yarn-error.log*
lerna-debug.log*
.pnpm-debug.log*

# Diagnostic reports (https://nodejs.org/api/report.html)

report.[0-9]_.[0-9]_.[0-9]_.[0-9]_.json

# Runtime data

pids
_.pid
_.seed
\*.pid.lock

# Directory for instrumented libs generated by jscoverage/JSCover

lib-cov

# Coverage directory used by tools like istanbul

coverage
\*.lcov

# nyc test coverage

.nyc_output

# Grunt intermediate storage (https://gruntjs.com/creating-plugins#storing-task-files)

.grunt

# Bower dependency directory (https://bower.io/)

bower_components

# node-waf configuration

.lock-wscript

# Compiled binary addons (https://nodejs.org/api/addons.html)

build/Release

# Dependency directories

node_modules/
jspm_packages/

# Snowpack dependency directory (https://snowpack.dev/)

web_modules/

# TypeScript cache

\*.tsbuildinfo

# Optional npm cache directory

.npm

# Optional eslint cache

.eslintcache

# Optional stylelint cache

.stylelintcache

# Microbundle cache

.rpt2_cache/
.rts2_cache_cjs/
.rts2_cache_es/
.rts2_cache_umd/

# Optional REPL history

.node_repl_history

# Output of 'npm pack'

\*.tgz

# Yarn Integrity file

.yarn-integrity

# dotenv environment variable files

.env
.env.development.local
.env.test.local
.env.production.local
.env.local

# parcel-bundler cache (https://parceljs.org/)

.cache
.parcel-cache

# Next.js build output

.next
out

# Nuxt.js build / generate output

.nuxt
dist

# Gatsby files

.cache/

# Comment in the public line in if your project uses Gatsby and not Next.js

# https://nextjs.org/blog/next-9-1#public-directory-support

# public

# vuepress build output

.vuepress/dist

# vuepress v2.x temp and cache directory

.temp
.cache

# Docusaurus cache and generated files

.docusaurus

# Serverless directories

.serverless/

# FuseBox cache

.fusebox/

# DynamoDB Local files

.dynamodb/

# TernJS port file

.tern-port

# Stores VSCode versions used for testing VSCode extensions

.vscode-test

# yarn v2

.yarn/cache
.yarn/unplugged
.yarn/build-state.yml
.yarn/install-state.gz
.pnp.\*

# IntelliJ based IDEs
.idea
19 changes: 16 additions & 3 deletions README.md
Original file line number Diff line number Diff line change
@@ -1,5 +1,18 @@
# Topsort.js
# topsort.js

This repository holds the official Topsort.js client library. It is built using TypeScript.

To install dependencies:

```bash
bun install
```

To run:

```bash
bun run src/index.ts
```


This repository holds the official Topsort.js client library. It is built using [TypeScript][typescript].

[typescript]: https://www.typescriptlang.org
Binary file added bun.lockb
Binary file not shown.
26 changes: 26 additions & 0 deletions package.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,26 @@
{
"name": "topsort.js",
"version": "1.0.0",
"description": "",
"private": true,
"packageManager": "[email protected]",
"main": "src/index.ts",
"author": "Márcio Barbosa <[email protected]>",
"license": "UNLICENSED",
"scripts": {
"build": "bun build",
"test": "bun test",
"lint": "bun lint",
"format": "bun format",
"prepublish": "bun prepublish",
"doctest": "bun run src/lib/doctest.test.ts"
},
"devDependencies": {
"@supabase/doctest-js": "^0.1.0",
"bun-types": "latest"
},
"peerDependencies": {
"typescript": "^5.0.0"
},
"dependencies": {}
}
40 changes: 40 additions & 0 deletions src/functions/report-event.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,40 @@
import { version } from "../../package.json";
import { Config, TopsortEvent } from "../interfaces/events.interface";

/**
* Reports an event to the Topsort API.
*
* @example
* ```js
* const event = { eventType: "test", eventData: {} };
* const config = { token: "my-token" };
* const result = await reportEvent(event, config);
* console.log(result); // { ok: true, retry: false }
* ```
*
* @param {TopsortEvent} e - The event to report.
* @param {Config} config - The configuration object containing URL and token.
* @returns {Promise<{ok: boolean, retry: boolean}>} The result of the report, indicating success and if a retry is needed.
*/
export async function reportEvent(e: TopsortEvent, config: Config): Promise<{ ok: boolean, retry: boolean }> {
try {
const url = (config.url || "https://api.topsort.com") + "/v2/events";
const r = await fetch(url, {
method: "POST",
headers: {
"Content-Type": "application/json",
// Can't use User-Agent header because of
// https://bugs.chromium.org/p/chromium/issues/detail?id=571722
"X-UA": `ts.js/${version}`,
Authorization: "Bearer " + config.token,
},
body: JSON.stringify(e),
// This parameter ensures in most browsers that the request is performed even in case the browser navigates to another page.
keepalive: true,
});
return { ok: r.ok, retry: r.status === 429 || r.status >= 500 };
} catch (error) {
console.error("Error reporting event:", error);
return { ok: false, retry: true };
}
}
2 changes: 2 additions & 0 deletions src/index.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
export * from './functions/report-event';
export * from './interfaces/events.interface';
52 changes: 52 additions & 0 deletions src/interfaces/events.interface.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,52 @@
interface Placement {
path: string;
}

export interface Entity {
type: "product";
id: string;
}

interface Impression {
resolvedBidId?: string;
entity?: Entity;
additionalAttribution?: Entity;
placement: Placement;
occurredAt: string;
opaqueUserId: string;
id: string;
}

interface Click {
resolvedBidId?: string;
entity?: Entity;
additionalAttribution?: Entity;
placement: Placement;
occurredAt: string;
opaqueUserId: string;
id: string;
}

interface Item {
productId: string;
quantity: number;
unitPrice: number;
}

interface Purchase {
occurredAt: string;
opaqueUserId: string;
id: string;
items: Item[];
}

export interface TopsortEvent {
impressions?: Impression[];
clicks?: Click[];
purchases?: Purchase[];
}

export interface Config {
token: string;
url?: string;
}
13 changes: 13 additions & 0 deletions src/lib/extract-comments.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
import * as fs from "fs";

export function extractJSDocComments(filePath: string): string[] {
const content = fs.readFileSync(filePath, "utf-8");
const comments = [];
const regex = /\/\*\*([\s\S]*?)\*\//g;
let match;
while ((match = regex.exec(content)) !== null) {
console.log("JSDoc Comment:", match[1]); // Log each extracted JSDoc comment
comments.push(match[1].trim());
}
return comments;
}
33 changes: 33 additions & 0 deletions src/lib/generate-test-cases.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,33 @@
import { extractJSDocComments } from "./extract-comments";

interface TestCase {
code: string;
expected: any;
}

export function generateTestCases(filePath: string): TestCase[] {
const comments = extractJSDocComments(filePath);
const testCases: TestCase[] = [];

comments.forEach((comment) => {
console.log(comment)
const exampleMatch = comment.match(/@example\s+\*?\s*```js([\s\S]*?)\s*\*?\s*```/);
if (exampleMatch) {
const exampleCode = exampleMatch[1].trim();
const expectedMatch = exampleCode.match(/console\.log\((.*)\);?\s*\/\/\s*(.*)/);
if (expectedMatch) {
try {
const expectedOutput = JSON.parse(expectedMatch[2].trim());
testCases.push({
code: exampleCode.replace(/console\.log\(.*\);/, ""),
expected: expectedOutput,
});
} catch (error) {
console.error("Failed to parse expected output:", expectedMatch[2].trim(), error);
}
}
}
});

return testCases;
}
Loading

0 comments on commit 58e9e45

Please sign in to comment.