{"id":13581143,"url":"https://github.com/campvanilla/casualdb","last_synced_at":"2025-10-20T17:49:03.782Z","repository":{"id":44364313,"uuid":"264432605","full_name":"campvanilla/casualdb","owner":"campvanilla","description":"Simple JSON \"database\" for Deno with type-safety! ⚡️","archived":false,"fork":false,"pushed_at":"2021-07-19T06:33:40.000Z","size":99,"stargazers_count":34,"open_issues_count":5,"forks_count":5,"subscribers_count":5,"default_branch":"main","last_synced_at":"2024-11-16T13:47:03.253Z","etag":null,"topics":["database","deno","json","prototyping"],"latest_commit_sha":null,"homepage":"","language":"TypeScript","has_issues":true,"has_wiki":null,"has_pages":null,"mirror_url":null,"source_name":null,"license":"mit","status":null,"scm":"git","pull_requests_enabled":true,"icon_url":"https://github.com/campvanilla.png","metadata":{"files":{"readme":"README.md","changelog":null,"contributing":".github/contributing.md","funding":null,"license":"LICENSE","code_of_conduct":null,"threat_model":null,"audit":null,"citation":null,"codeowners":null,"security":null,"support":null}},"created_at":"2020-05-16T12:32:01.000Z","updated_at":"2023-12-26T05:20:04.000Z","dependencies_parsed_at":"2022-08-29T21:41:30.125Z","dependency_job_id":null,"html_url":"https://github.com/campvanilla/casualdb","commit_stats":null,"previous_names":[],"tags_count":5,"template":false,"template_full_name":null,"repository_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repositories/campvanilla%2Fcasualdb","tags_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repositories/campvanilla%2Fcasualdb/tags","releases_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repositories/campvanilla%2Fcasualdb/releases","manifests_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repositories/campvanilla%2Fcasualdb/manifests","owner_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/owners/campvanilla","download_url":"https://codeload.github.com/campvanilla/casualdb/tar.gz/refs/heads/main","host":{"name":"GitHub","url":"https://github.com","kind":"github","repositories_count":247445652,"owners_count":20939952,"icon_url":"https://github.com/github.png","version":null,"created_at":"2022-05-30T11:31:42.601Z","updated_at":"2022-07-04T15:15:14.044Z","host_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub","repositories_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repositories","repository_names_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repository_names","owners_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/owners"}},"keywords":["database","deno","json","prototyping"],"created_at":"2024-08-01T15:01:58.498Z","updated_at":"2025-10-20T17:49:03.392Z","avatar_url":"https://github.com/campvanilla.png","language":"TypeScript","funding_links":[],"categories":["TypeScript"],"sub_categories":[],"readme":"\u003cp align=\"center\"\u003e\n  \u003cimg src=\"https://user-images.githubusercontent.com/6426069/82755043-bb65a700-9dee-11ea-9de4-e57476f216db.png\" width=\"300\" /\u003e\n\u003c/p\u003e\n\n\u003cp align=\"center\"\u003e\n  \u003csub\u003e\n    Simple JSON \"database\" for Deno with type-safety! ⚡️\n  \u003c/sub\u003e\n\u003c/p\u003e\n\n\u003cp align=\"center\"\u003e\n  \u003cstrong\u003eWARNING\u003c/strong\u003e: This project is still in beta phase. We are actively working on enhancing the API and ironing out kinks. If you find a bug or have a feature request, feel free to create an issue or contribute. 🙂\n\u003c/p\u003e\n\n![GitHub release (latest SemVer including pre-releases)](https://img.shields.io/github/v/release/campvanilla/casualdb?color=%232ecc71\u0026include_prereleases\u0026style=flat-square) \u003c!-- ALL-CONTRIBUTORS-BADGE:START - Do not remove or modify this section --\u003e\n[![All Contributors](https://img.shields.io/badge/all_contributors-5-orange.svg?style=flat-square)](#contributors-)\n\u003c!-- ALL-CONTRIBUTORS-BADGE:END --\u003e\n\n## Contents\n\n* [Quick Usage](#quick-usage)\n* [Installation](#installation)\n* [API](#api)\n* [Inspiration](#inspiration)\n  * [Disclaimer](#disclaimer) ⚠️\n* [Contributing](#contributing)\n\n## Quick Usage\n\n``` ts\n// create an interface to describe the structure of your JSON\ninterface Schema {\n  posts: Array\u003c{\n    id: number;\n    title: string;\n    views: number;\n  }\u003e;\n  user: {\n    name: string;\n  };\n}\n\nconst db = new CasualDB\u003cSchema\u003e(); // instantiate the db, casually 🤓\nawait db.connect(\"./test-db.json\"); // \"connect\" to the db (JSON file)\n\n// (optional) seed it with data, if starting with an empty db\nawait db.seed({\n  posts: [\n    { id: 1, title: \"Post 1\", views: 99 },\n    { id: 2, title: \"Post 2\", views: 30 },\n  ],\n  user: { name: \"Camp Vanilla\" },\n});\n\nconst posts = await db.get\u003cSchema['posts']\u003e('posts'); // pass the interface key in order for type-checking to work\n\nconst postTitlesByViews = (\n  posts\n    .sort(['views']) // sort by views (ascending)\n    .pick(['title']) // pick the title of every post\n    .value() // =\u003e ['Post 2', 'Post 1']\n);\n```\n\n## Installation\n\n``` ts\nimport { CasualDB } from \"https://deno.land/x/casualdb@0.1.4/mod.ts\";\n\n// create an interface to describe the structure of your JSON\ninterface Schema {\n  posts: Array\u003c{\n    id: number;\n    title: string;\n    views: number;\n  }\u003e;\n  user: {\n    name: string;\n  };\n}\n\nconst db = new CasualDB\u003cSchema\u003e();\n```\n\nNote: When running via deno, this module will require you to pass the following flags (all flags are mandatory):-\n\n* `--allow-read` : in order to be able to **read** the JSON files\n* `--allow-write`: in order to be able to **write** to the JSON files\n* `--unstable`   : this module uses the experimental Worker API in deno, and hence requires this flag\n* `--allow-net`  : this is to enable to download of the Worker file.\n\nIf you want to always run the latest code (from the `master` branch) of this module, install via:\n```ts\nimport { CasualDB } from \"https://deno.land/x/casualdb/mod.ts\";\n```\n\n## API\n\n### new CasualDB\u003cSchema\u003e()\n\nReturns an instance of the _CasualDB_. Passing in a interface describing your JSON data ensures that **type checking works correctly**. The following are the methods available on this class instance\n\n* [.connect()](#casual-db-connect)\n* [.get()](#casual-db-get)\n* [.seed()](#casual-db-seed)\n* [.write()](#casual-db-write)\n\n\u003ch4 id='casual-db-connect'\u003e\n  \u003ccode\u003e.connect(pathToJsonFile: string, options?: ConnectOptions)\u003c/code\u003e\n\u003c/h4\u003e\n\nCreates a _connection_ to a json file passed as parameter. Returns a promise.\n\nConnectOptions:\n\n  + `bailIfNotPresent` \u003cBoolean\u003e: Controls whether you would like an error to be thrown if the file being connected to does not exist. Default = `false` .\n\n``` ts\nawait db.connect(\"./test-db.json\");\n\n// or with options\n\nawait db.connect(\"./test-db.json\", {\n  bailIfNotPresent: true,\n});\n```\n\n\u003ch4 id='casual-db-get'\u003e\n  \u003ccode\u003e.get\u003cT\u003e(jsonPath: string)\u003c/code\u003e\n\u003c/h4\u003e\n\nFetches value from connected JSON file. Takes an object _path_ as parameter. Returns a `Promise\u003cCollectionOperator | PrimitiveOperator\u003e` .\n**Important**: For type checking to work, ensure that the Template Type \u003cT\u003e is provided to `.get\u003cT\u003e()` . If this is not provided, typescript cannot decide a _CollectionOperator_ or _PrimitiveOperator_ has been returned and hence you'd have to manually narrow it down for TS.\n\n``` ts\ninterface Schema {\n  posts: Array\u003c{\n    id: number;\n    title: string;\n    views: number;\n  }\u003e;\n  user: {\n    name: string;\n  };\n}\n\nawait db.get\u003cSchema[\"posts\"]\u003e('posts'); // Returns a Promise\u003cCollectionOperator\u003e\n\n// or\n\nawait db.get\u003cSchema[\"posts\"][number][\"id\"]\u003e('posts.0.id'); // Returns a Promise\u003cPrimitiveOperator\u003e\n```\n\n\u003ch4 id=\"casual-db-seed\"\u003e\n  \u003ccode\u003e.seed(data: Schema)\u003c/code\u003e\n\u003c/h4\u003e\n\nOverrides the contents of the connected JSON file. This is beneficial for when you don't already have data in the file or you want to add some defaults. Returns a promise.\n\n``` ts\ninterface Schema {\n  posts: Array\u003c{\n    id: number;\n    title: string;\n    views: number;\n  }\u003e;\n  user: {\n    name: string;\n  };\n}\n\nawait db.seed({\n  posts: [\n    { id: 1, title: \"Post 1\", views: 99 },\n    { id: 2, title: \"Post 2\", views: 30 },\n  ],\n  user: { name: \"Camp Vanilla\" },\n});\n```\n\n\u003ch4 id='casual-db-write'\u003e\n  \u003ccode\u003e.write(jsonPath: string, data: any)\u003c/code\u003e\n\u003c/h4\u003e\n\nWrites the provided value to the Object path provided. Returns a promise.\n\n``` ts\nawait db.write('posts', [\n  { id: 1, title: \"Post 1\", views: 99 },\n  { id: 2, title: \"Post 2\", views: 30 },\n]);\n\n// or\n\nawait db.write('posts.0.title', 'Post 1');\n```\n\n### PrimitiveOperator\n\nWhen performing a `db.get()` on a path that returns a non-array value, the Promise resolves to an instance of `PrimitiveOperator` . The _PrimitiveOperator_ class encapsulates functions that allow you work with any non-array-like data in javascript (eg. `object` , `string` , `number` , `boolean` ). All functions that are a part of _PrimitiveOperator_ allow function chaining.\n\n``` ts\ninterface Schema {\n  posts: Array\u003c{\n    id: number;\n    title: string;\n    views: number;\n  }\u003e;\n  user: {\n    name: string;\n  };\n}\n\nconst data = await db.get\u003cSchema[\"posts\"]\u003e('posts'); // ❌ Not a PrimitiveOperator as the value is going to be an array\n\nconst data = await db.get\u003cSchema[\"posts\"][number]\u003e('posts.0'); // ✅ PrimitiveOperator as the value is a non-array.\n```\n\nInstances of this class have the following methods:\n\n* [.value()](#primitive-operator-value)\n* [.update()](#primitive-operator-update)\n* [.pick()](#primitive-operator-pick)\n\n\u003ch4 id='primitive-operator-value'\u003e\n  \u003ccode\u003e.value()\u003c/code\u003e\n\u003c/h4\u003e\n\nReturns the value of the data.\n\n``` ts\nconst data = await db.get\u003cSchema[\"posts\"][number]\u003e('posts.0');\n\ndata.value(); // { id: 1, title: \"Post 1\", views: 99 }\n```\n\n\u003ch4 id='primitive-operator-update'\u003e\n  \u003ccode\u003e.update\u003cT\u003e(updateMethod: (currentValue) =\u003e T)\u003c/code\u003e\n\u003c/h4\u003e\n\nMethod to update the data. Method takes an updater-function as parameter. The updater-function will receive the value you want to update and expects a return value. The type of the updated data is inferred by the ReturnType of the updater-function.\n\n``` ts\nconst data = await db.get\u003cSchema[\"posts\"][number]\u003e('posts.0');\n\ndata\n  .update((value) =\u003e ({\n    title: \"Modified Post\",\n  }))\n  .value(); // { id: 1, title: \"Modified Post\" }\n```\n\n\u003ch4 id='primitive-operator-pick'\u003e\n  .pick(keys: string[])\n\u003c/h4\u003e\n\nPicks and returns a subset of keys from the data. Method allows only keys present on data. If the data is not an object, method returns the data as is.\n\n``` ts\nconst data = await db.get\u003cSchema[\"posts\"][number]\u003e('posts.0');\n\ndata\n  .pick([\"id\", \"title\"])\n  .value(); // { id: 1, title: \"Post 1\" }\n```\n\n### CollectionOperator\n\nWhen performing a `db.get()` on a path that returns an array, the Promise resolves to a instance of `CollectionOperator` . The _CollectionOperator_ class encapsulates functions that allow you work with array-like data (collection of items). All functions that are a part of _CollectionOperator_ allow function chaining.\n\n``` ts\ninterface Schema {\n  posts: Array\u003c{\n    id: number;\n    title: string;\n    views: number;\n  }\u003e;\n  user: {\n    name: string;\n  };\n}\n\nconst data = await db.get\u003cSchema[\"posts\"]\u003e('posts'); // ✅ CollectionOperator as the value is an array.\n\nconst data = await db.get\u003cSchema[\"posts\"][number]\u003e('posts.0'); // ❌ PrimitiveOperator as the value is a non-array.\n```\n\nInstances of this class contain the following methods. All methods are chainable:\n\n* [.value()](#collection-operator-value)\n* [.size()](#collection-operator-size)\n* [.findOne()](#collection-operator-findOne)\n* [.findAllAndUpdate()](#collection-operator-findAllAndUpdate)\n* [.findAllAndRemove()](#collection-operator-findAllAndRemove)\n* [.findById()](#collection-operator-findById)\n* [.findByIdAndRemove()](#collection-operator-findByIdAndRemove)\n* [.findByIdAndUpdate()](#collection-operator-findByIdAndUpdate)\n* [.sort()](#collection-operator-sort)\n* [.page()](#collection-operator-page)\n* [.pick()](#collection-operator-pick)\n\n\u003ch4 id='collection-operator-value'\u003e\n  \u003ccode\u003e.value()\u003c/code\u003e\n\u003c/h4\u003e\n\nReturns the value of the data.\n\n``` ts\nconst data = await db.get\u003cSchema[\"posts\"]\u003e('posts');\n\nconsole.log(data.value()); // [ { id: 1, title: \"Post 1\", views: 99 }, { id: 2, title: \"Post 2\", views: 30 }, ]\n```\n\n\u003ch4 id='collection-operator-size'\u003e\n  \u003ccode\u003e.size()\u003c/code\u003e\n\u003c/h4\u003e\n\nReturns the length of the data.\n\n``` ts\nconst data = await db.get\u003cSchema[\"posts\"]\u003e('posts');\n\nconsole.log(data.size()); // 2\n```\n\n\u003ch4 id='collection-operator-findOne'\u003e\u003ccode\u003e\n.findOne(predicate: Object | Function =\u003e boolean)\n\u003c/code\u003e\u003c/h4\u003e\n\nSearches through the collection items and returns an item if found, else returns an instance of `PrimitiveOperator\u003cnull\u003e` . The predicate can be of two forms:\n\n1. An object with keys that you would like to match. The keys of the object should be a subset of the keys available on the items of the collection.\n2. A search-function where you can provide your custom logic and return `true` for the condition you are looking for.\n\nReturns a `PrimitiveOperator` or `CollectionOperator` based on type of the found element.\n\n``` ts\nconst data = await db.get\u003cSchema[\"posts\"]\u003e('posts');\n\ndata\n  .findOne({ id: 1 })\n  .value();// { id: 1, title: \"Post 1\", views: 99 }\n\n// or\n\ndata\n  .findOne((value) =\u003e {\n    return value.id === 1\n  })\n  .value(); // { id: 1, title: \"Post 1\", views: 99 }\n```\n\n\u003ch4 id='collection-operator-push'\u003e\u003ccode\u003e\n.push(value)\n\u003c/code\u003e\u003c/h4\u003e\n\nPush a new value into the collection. Returns a `CollectionOperator` with the updated items.\n\n``` ts\nconst data = await db.get\u003cSchema[\"posts\"]\u003e('posts');\n\ndata\n  .push({ id: 3, post: 'Post 3', views: 45 })\n  .value(); // [ { id: 1, title: \"Post 1\", views: 99 }, { id: 2, title: \"Post 2\", views: 30 }, { id: 3, title: \"Post 3\", views: 45 } ]\n```\n\n\u003ch4 id='collection-operator-findAll'\u003e\u003ccode\u003e\n.findAll(predicate: Object | Function =\u003e boolean)\n\u003c/code\u003e\u003c/h4\u003e\n\nSearches through the items of the collection and returns a `CollectionOperator` of all occurrences that satisfy the predicate. The predicate can be of two forms:\n\n1. An object with keys that you would like to match. The keys of the object should be a subset of the keys available on the items of the collection.\n2. A search-function where you can provide your custom logic and return `true` for the condition you are looking for.\n\nReturns a `CollectionOperator` with the subset of items.\n\n``` ts\nconst data = await db.get\u003cSchema[\"posts\"]\u003e('posts');\n\ndata\n  .findAll({ title: 'Post 1' })\n  .value();// [{ id: 1, title: \"Post 1\", views: 99 }]\n\n// or\n\ndata\n  .findAll((value) =\u003e {\n    return value.views \u003e 40;\n  })\n  .value(); // [{ id: 1, title: \"Post 1\", views: 99 },{ id: 3, title: \"Post 3\", views: 45 }];\n```\n\n\u003ch4 id='collection-operator-findAllAndUpdate'\u003e\u003ccode\u003e\n.findAllAndUpdate(predicate: Object | Function =\u003e boolean, updateMethod: (value) =\u003e T)\n\u003c/code\u003e\u003c/h4\u003e\n\nSearches through the collection and returns a `CollectionOperator` with all occurrences that satisfy the predicate updated with the return value of the _updateMethod_. The predicate can be of two forms:\n\n1. An object with keys that you would like to match. The keys of the object should be a subset of the keys available on the items of the collection.\n2. A search-function where you can provide your custom logic and return `true` for the condition you are looking for.\n\nReturns a `CollectionOperator` with the updated array.\n\n``` ts\nconst data = await db.get\u003cSchema[\"posts\"]\u003e('posts');\n\ndata\n  .findAllAndUpdate({ title: 'Post 1' }, (value) =\u003e ({ ...value, title: 'Modified Post' }))\n  .value(); // [{ id: 1, title: \"Modified Post\", views: 99 },{ id: 2, title: \"Post 2\", views: 30 }, { id: 3, title: \"Post 3\", views: 45 }]\n\n// or\n\ndata\n  .findAllAndUpdate((value) =\u003e {\n    return value.views \u003e 40;\n  }, (value) =\u003e ({\n    ...value,\n    title: 'Trending Post'\n  }))\n  .value(); // [{ id: 1, title: \"Trending Post\", views: 99 }, { id: 2, title: \"Post 2\", views: 30 }, { id: 3, title: \"Trending Post\", views: 45 }];\n```\n\n\u003ch4 id='collection-operator-findAllAndRemove'\u003e\u003ccode\u003e\n.findAllAndRemove(predicate: Object | Function =\u003e boolean, updateMethod: (value) =\u003e T)\n\u003c/code\u003e\u003c/h4\u003e\n\nSearches through the collection and returns a new `CollectionOperator` where all occurrences that satisfy the predicate are *omitted*. The predicate can be of two forms:\n\n1. An object with keys that you would like to match. The keys of the object should be a subset of the keys available on the items of the collection.\n2. A search-function where you can provide your custom logic and return `true` for the condition you are looking for.\n\nReturns a `CollectionOperator` with the updated array.\n\n``` ts\nconst data = await db.get\u003cSchema[\"posts\"]\u003e('posts');\n\ndata\n  .findAllAndRemove({ title: 'Post 1' })\n  .value(); // [{ id: 2, title: \"Post 2\", views: 30 }, { id: 3, title: \"Post 3\", views: 45 }]\n\n// or\n\ndata\n  .findAllAndRemove((value) =\u003e value.views \u003e 40)\n  .value(); // [{ id: 2, title: \"Post 2\", views: 30 }];\n```\n\n\u003ch4 id=\"collection-operator-findById\"\u003e\n\u003ccode\u003e.findById(id: string)\u003c/code\u003e\n\u003c/h4\u003e\n\nSyntactical sugar for `.findOne({ id })` .\n\n\u003ch4 id=\"collection-operator-findByIdAndRemove\"\u003e\n\u003ccode\u003e.findByIdAndRemove(id: string)\u003c/code\u003e\n\u003c/h4\u003e\n\nSyntactical sugar for `.findAllAndRemove({ id })` .\n\n\u003ch4 id=\"collection-operator-findByIdAndUpdate\"\u003e\n\u003ccode\u003e.findByIdAndUpdate(id: string, updateMethod: (value) =\u003e T)\u003c/code\u003e\n\u003c/h4\u003e\n\nSyntactical sugar for `.findAllAndUpdate({ id }, updateMethod)` .\n\n\u003ch4 id=\"collection-operator-sort\"\u003e\n\u003ccode\u003e.sort(predicate: string[] | Function =\u003e boolean)\u003c/code\u003e\n\u003c/h4\u003e\n\nSorts and returns a new sorted `CollectionOperator` instance. The comparison predicate can be one of two types:\n\n* **an array of keys** to select for sorting the items in the collection (priority is left-right).\u003cbr /\u003e\n  For example, when the predicate is `['views','id']` , the method will first sort *posts* in ascending order of *views* that each post has. Any posts which have the *same* number of views, will then be sorted by `id` .\n* a **compare function** similar to [ `Array.prototype.sort` ](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/sort#Parameters)'s `compareFunction` .\n\n``` ts\nconst posts = await db.get\u003cSchema[\"posts\"]\u003e('posts');\n\nposts\n  .sort(['views'])\n  .value() // [{ id: 2, title: \"Post 2\", views: 30 }, { id: 1, title: \"Post 1\", views: 99 }]\n\n// or\n\nposts\n  .sort((a,b) =\u003e a.views - b.views)\n  .value() // [{ id: 2, title: \"Post 2\", views: 30 }, { id: 1, title: \"Post 1\", views: 99 }]\n```\n\n\u003ch4 id=\"collection-operator-page\"\u003e\n\u003ccode\u003e.page(page: number, pageSize: number)\u003c/code\u003e\n\u003c/h4\u003e\n\nReturns a paginated subset of the collection.\n\n``` ts\nconst posts = await db.get\u003cSchema[\"posts\"]\u003e('posts');\n\nposts\n  .page(1, 1)\n  .value() // [{ id: 1, title: \"Post 1\", views: 99 }]\n```\n\n\u003ch4 id=\"collection-operator-pick\"\u003e\n\u003ccode\u003e.pick(keys: string[])\u003c/code\u003e\n\u003c/h4\u003e\n\nReturns a `CollectionOperator` of items with each item having only the *picked* keys. Only keys present on the type of the items in the collection are allowed. If the item is not an object, this method returns an empty object ( `{}` ) for it.\n\n``` ts\nconst posts = await db.get\u003cSchema[\"posts\"]\u003e('posts');\n\nposts\n  .pick(['title'])\n  .value() // [{ title: \"Post 1\" }, { title: \"Post 2\" }]\n```\n\n## Inspiration\n\nThis project has taken inspiration from [lowdb](https://github.com/typicode/lowdb) for the concept and [mongoose](https://mongoosejs.com/) for certain parts of the `CollectionOperator` API.\n\nIt aims to simplify the process of setting up a full-fledged db when building prototypes or small-scale applications like CLI tools or toy apps for Deno.\n\n\n### 🚧 ⚠️ Disclaimer ⚠️ 🚧\n\n\u003ca id=\"disclaimer\"\u003e\u003c/a\u003e\n\n**Disclaimer** : As mentioned above, this module is best used for small-scale apps and should not be used in a large production application and you may face issues like:\n* concurrency management (for writes)\n* storing and parsing large amounts of JSON data.\n\n## Contributing\n\nWant to raise an issue or pull request? Do give our [Contribution Guidelines](./.github/CONTRIBUTING.md) page a read. 🤓\n\n## Contributors ✨\n\nThanks goes to these wonderful people ([emoji key](https://allcontributors.org/docs/en/emoji-key)):\n\n\u003c!-- ALL-CONTRIBUTORS-LIST:START - Do not remove or modify this section --\u003e\n\u003c!-- prettier-ignore-start --\u003e\n\u003c!-- markdownlint-disable --\u003e\n\u003ctable\u003e\n  \u003ctr\u003e\n    \u003ctd align=\"center\"\u003e\u003ca href=\"https://abinavseelan.com/?utm_source=github\u0026utm_medium=documentation-allcontributors\u0026utm_content=casualdb\"\u003e\u003cimg src=\"https://avatars2.githubusercontent.com/u/6417910?v=4?s=100\" width=\"100px;\" alt=\"\"/\u003e\u003cbr /\u003e\u003csub\u003e\u003cb\u003eAbinav Seelan\u003c/b\u003e\u003c/sub\u003e\u003c/a\u003e\u003cbr /\u003e\u003ca href=\"https://github.com/campvanilla/casualdb/commits?author=abinavseelan\" title=\"Code\"\u003e💻\u003c/a\u003e \u003ca href=\"https://github.com/campvanilla/casualdb/commits?author=abinavseelan\" title=\"Documentation\"\u003e📖\u003c/a\u003e \u003ca href=\"#ideas-abinavseelan\" title=\"Ideas, Planning, \u0026 Feedback\"\u003e🤔\u003c/a\u003e \u003ca href=\"https://github.com/campvanilla/casualdb/commits?author=abinavseelan\" title=\"Tests\"\u003e⚠️\u003c/a\u003e\u003c/td\u003e\n    \u003ctd align=\"center\"\u003e\u003ca href=\"https://aditimohanty.com/?utm_source=github\u0026utm_medium=documentation-allcontributors\u0026utm_content=casualdb\"\u003e\u003cimg src=\"https://avatars3.githubusercontent.com/u/6426069?v=4?s=100\" width=\"100px;\" alt=\"\"/\u003e\u003cbr /\u003e\u003csub\u003e\u003cb\u003eAditi Mohanty\u003c/b\u003e\u003c/sub\u003e\u003c/a\u003e\u003cbr /\u003e\u003ca href=\"https://github.com/campvanilla/casualdb/commits?author=rheaditi\" title=\"Code\"\u003e💻\u003c/a\u003e \u003ca href=\"https://github.com/campvanilla/casualdb/commits?author=rheaditi\" title=\"Documentation\"\u003e📖\u003c/a\u003e \u003ca href=\"#ideas-rheaditi\" title=\"Ideas, Planning, \u0026 Feedback\"\u003e🤔\u003c/a\u003e \u003ca href=\"https://github.com/campvanilla/casualdb/commits?author=rheaditi\" title=\"Tests\"\u003e⚠️\u003c/a\u003e\u003c/td\u003e\n    \u003ctd align=\"center\"\u003e\u003ca href=\"http://www.willterry.me\"\u003e\u003cimg src=\"https://avatars0.githubusercontent.com/u/12277149?v=4?s=100\" width=\"100px;\" alt=\"\"/\u003e\u003cbr /\u003e\u003csub\u003e\u003cb\u003eWilliam Terry\u003c/b\u003e\u003c/sub\u003e\u003c/a\u003e\u003cbr /\u003e\u003ca href=\"https://github.com/campvanilla/casualdb/issues?q=author%3ATezza48\" title=\"Bug reports\"\u003e🐛\u003c/a\u003e\u003c/td\u003e\n    \u003ctd align=\"center\"\u003e\u003ca href=\"http://yaofur.com/\"\u003e\u003cimg src=\"https://avatars0.githubusercontent.com/u/289392?v=4?s=100\" width=\"100px;\" alt=\"\"/\u003e\u003cbr /\u003e\u003csub\u003e\u003cb\u003eKeith Yao\u003c/b\u003e\u003c/sub\u003e\u003c/a\u003e\u003cbr /\u003e\u003ca href=\"https://github.com/campvanilla/casualdb/issues?q=author%3Akebot\" title=\"Bug reports\"\u003e🐛\u003c/a\u003e \u003ca href=\"https://github.com/campvanilla/casualdb/commits?author=kebot\" title=\"Code\"\u003e💻\u003c/a\u003e\u003c/td\u003e\n    \u003ctd align=\"center\"\u003e\u003ca href=\"https://github.com/jackfiszr\"\u003e\u003cimg src=\"https://avatars.githubusercontent.com/u/7147395?v=4?s=100\" width=\"100px;\" alt=\"\"/\u003e\u003cbr /\u003e\u003csub\u003e\u003cb\u003eJacek Fiszer\u003c/b\u003e\u003c/sub\u003e\u003c/a\u003e\u003cbr /\u003e\u003ca href=\"https://github.com/campvanilla/casualdb/commits?author=jackfiszr\" title=\"Code\"\u003e💻\u003c/a\u003e\u003c/td\u003e\n  \u003c/tr\u003e\n\u003c/table\u003e\n\n\u003c!-- markdownlint-restore --\u003e\n\u003c!-- prettier-ignore-end --\u003e\n\n\u003c!-- ALL-CONTRIBUTORS-LIST:END --\u003e\n\nThis project follows the [all-contributors](https://github.com/all-contributors/all-contributors) specification. Contributions of any kind welcome!\n","project_url":"https://awesome.ecosyste.ms/api/v1/projects/github.com%2Fcampvanilla%2Fcasualdb","html_url":"https://awesome.ecosyste.ms/projects/github.com%2Fcampvanilla%2Fcasualdb","lists_url":"https://awesome.ecosyste.ms/api/v1/projects/github.com%2Fcampvanilla%2Fcasualdb/lists"}