summaryrefslogtreecommitdiffhomepage
path: root/js/src/util
diff options
context:
space:
mode:
authorDax Raad <[email protected]>2025-05-17 21:31:42 -0400
committerDax Raad <[email protected]>2025-05-26 12:40:17 -0400
commita34d020bc6b252e842f042d935c7a0e6444460cf (patch)
treeea3484499dff80e82d421e879ab639133ae9c3b4 /js/src/util
parent96fbc37f0175052291f8a096d530bd4480f6cb19 (diff)
downloadopencode-a34d020bc6b252e842f042d935c7a0e6444460cf.tar.gz
opencode-a34d020bc6b252e842f042d935c7a0e6444460cf.zip
sync
Diffstat (limited to 'js/src/util')
-rw-r--r--js/src/util/context.ts25
-rw-r--r--js/src/util/log.ts27
2 files changed, 52 insertions, 0 deletions
diff --git a/js/src/util/context.ts b/js/src/util/context.ts
new file mode 100644
index 000000000..ec686293e
--- /dev/null
+++ b/js/src/util/context.ts
@@ -0,0 +1,25 @@
+import { AsyncLocalStorage } from "node:async_hooks";
+
+export namespace Context {
+ export class NotFound extends Error {
+ constructor(public readonly name: string) {
+ super(`No context found for ${name}`);
+ }
+ }
+
+ export function create<T>(name: string) {
+ const storage = new AsyncLocalStorage<T>();
+ return {
+ use() {
+ const result = storage.getStore();
+ if (!result) {
+ throw new NotFound(name);
+ }
+ return result;
+ },
+ provide<R>(value: T, fn: () => R) {
+ return storage.run<R>(value, fn);
+ },
+ };
+ }
+}
diff --git a/js/src/util/log.ts b/js/src/util/log.ts
new file mode 100644
index 000000000..9de4eb495
--- /dev/null
+++ b/js/src/util/log.ts
@@ -0,0 +1,27 @@
+export namespace Log {
+ export function create(tags?: Record<string, any>) {
+ tags = tags || {};
+
+ const result = {
+ info(message?: any, extra?: Record<string, any>) {
+ const prefix = Object.entries({
+ ...tags,
+ ...extra,
+ })
+ .map(([key, value]) => `${key}=${value}`)
+ .join(" ");
+ console.log(prefix, message);
+ return result;
+ },
+ tag(key: string, value: string) {
+ if (tags) tags[key] = value;
+ return result;
+ },
+ clone() {
+ return Log.create({ ...tags });
+ },
+ };
+
+ return result;
+ }
+}