Magic IndexedDB – Universal Translation Layer

Author: Lance Wright II

Published: March 26, 2025

Sorry Image Not Found!

Magic IndexedDB – Universal Translation Layer

Bringing true LINQ to IndexedDB to any framework, language, or platform!


1. What is the Universal Translation Layer?

Magic IndexedDB isn’t just a Blazor library—it’s a fully universal LINQ to IndexedDB translation system that any language or framework can hook into.

Instead of directly wrapping IndexedDB APIs, Magic IndexedDB acts as a middleware layer, translating LINQ-based queries into optimized IndexedDB requests using Dexie.js as the underlying IndexedDB wrapper.

Why Use Dexie.js?

Mature & well-supported library

Optimized IndexedDB access

Battle-tested by thousands of developers

Prevents reinventing the wheel—Magic IndexedDB focuses on the translation, not the direct API

🚀 The Goal: Build a universal IndexedDB querying system any language can implement.


2. How Queries Are Translated

Magic IndexedDB converts LINQ-style operations into a structured, universal query format, optimizing IndexedDB interactions wherever possible. However, due to IndexedDB's inherent limitations, certain operations must be handled using cursors instead of direct indexed queries.


📌 Supported Query Operators
🧮 Basic Comparison
Operator JavaScript Key Description IndexedDB Optimized?
==EqualExact match✅ Yes
!=NotEqualNot equal✅ Yes
>GreaterThanGreater than✅ Yes
>=GreaterThanOrEqualGreater than or equal✅ Yes
<LessThanLess than✅ Yes
<=LessThanOrEqualLess than or equal✅ Yes

🔤 String Matching
Operator JavaScript Key Description IndexedDB Optimized?
.startsWith()StartsWithString starts with✅ Yes* (case-insensitive = 🚫)
!x.startsWith()NotStartsWithDoes not start with🚫 Cursor Required
.endsWith()EndsWithString ends with🚫 Cursor Required
!x.endsWith()NotEndsWithDoes not end with🚫 Cursor Required
.contains()ContainsString/array contains🚫 Cursor Required
!x.contains()NotContainsDoes not contain🚫 Cursor Required

🚨 Important Notes:

Indexed queries are optimized, but operations like .Contains() always require a cursor scan.

StartsWith() is indexable if the comparison is case-sensitive (caseSensitive: true in the universal layer).

⚠️ If StringComparison.OrdinalIgnoreCase or caseSensitive: false is used, cursor fallback will occur.

🧪 This may not be everything—validate against QUERY_OPERATIONS in queryConstants.js.


📚 Array & Length Operations
Operator JavaScript Key Description IndexedDB Optimized?
.In([a, b, c])InMatches any value in array✅ Yes
.length == XLengthEqualLength equals X🚫 Cursor Required
.length != XNotLengthEqualLength not equal to X🚫 Cursor Required
.length > XLengthGreaterThanLength greater than X🚫 Cursor Required
.length >= XLengthGreaterThanOrEqualGreater or equal to X🚫 Cursor Required
.length < XLengthLessThanLength less than X🚫 Cursor Required
.length <= XLengthLessThanOrEqualLess than or equal to X🚫 Cursor Required

📅 Date Operations
Operator JavaScript Key IndexedDB Optimized?
x.Month == 7MonthEqual🚫 Cursor Required
x.Month != 7NotMonthEqual🚫 Cursor Required
x.Month > 7MonthGreaterThan🚫 Cursor Required
x.Month >= 7MonthGreaterThanOrEqual🚫 Cursor Required
x.Month < 7MonthLessThan🚫 Cursor Required
x.Month <= 7MonthLessThanOrEqual🚫 Cursor Required
x.Day == 15DayEqual🚫 Cursor Required
x.Day != 15NotDayEqual🚫 Cursor Required
x.Day > 15DayGreaterThan🚫 Cursor Required
x.Day >= 15DayGreaterThanOrEqual🚫 Cursor Required
x.Day < 15DayLessThan🚫 Cursor Required
x.Day <= 15DayLessThanOrEqual🚫 Cursor Required
x.DayOfWeek == XDayOfWeekEqual🚫 Cursor Required
x.DayOfWeek != XNotDayOfWeekEqual🚫 Cursor Required
x.DayOfWeek > XDayOfWeekGreaterThan🚫 Cursor Required
x.DayOfWeek >= XDayOfWeekGreaterThanOrEqual🚫 Cursor Required
x.DayOfWeek < XDayOfWeekLessThan🚫 Cursor Required
x.DayOfWeek <= XDayOfWeekLessThanOrEqual🚫 Cursor Required
x.DayOfYear == 128DayOfYearEqual🚫 Cursor Required
x.DayOfYear != 128NotDayOfYearEqual🚫 Cursor Required
x.DayOfYear > 128DayOfYearGreaterThan🚫 Cursor Required
x.DayOfYear >= 128DayOfYearGreaterThanOrEqual🚫 Cursor Required
x.DayOfYear < 128DayOfYearLessThan🚫 Cursor Required
x.DayOfYear <= 128DayOfYearLessThanOrEqual🚫 Cursor Required
x.Year == 2024YearEqual✅ Yes
x.Year != 2024NotYearEqual🚫 Cursor Required
x.Year > 2024YearGreaterThan✅ Yes
x.Year >= 2024YearGreaterThanOrEqual✅ Yes
x.Year < 2024YearLessThan✅ Yes
x.Year <= 2024YearLessThanOrEqual✅ Yes

🧪 Type Checks
Operator JavaScript Key Description IndexedDB Optimized?
typeof x === "number"TypeOfNumberValue is a number🚫 Cursor Required
typeof x === "string"TypeOfStringValue is a string🚫 Cursor Required
x instanceof DateTypeOfDateValue is a valid Date🚫 Cursor Required
Array.isArray(x)TypeOfArrayValue is an array🚫 Cursor Required
typeof x === "object"TypeOfObjectValue is a plain object🚫 Cursor Required
x instanceof BlobTypeOfBlobValue is a Blob🚫 Cursor Required
x instanceof ArrayBufferTypeOfArrayBufferValue is an ArrayBuffer or typed array🚫 Cursor Required
x instanceof FileTypeOfFileValue is a File🚫 Cursor Required
!(typeof x === "number")NotTypeOfNumberValue is not a number🚫 Cursor Required
!(typeof x === "string")NotTypeOfStringValue is not a string🚫 Cursor Required
!(x instanceof Date)NotTypeOfDateValue is not a valid Date🚫 Cursor Required
!Array.isArray(x)NotTypeOfArrayValue is not an array🚫 Cursor Required
!(typeof x === "object")NotTypeOfObjectValue is not a plain object🚫 Cursor Required
!(x instanceof Blob)NotTypeOfBlobValue is not a Blob🚫 Cursor Required
!(x instanceof ArrayBuffer)NotTypeOfArrayBufferValue is not an ArrayBuffer or typed array🚫 Cursor Required
!(x instanceof File)NotTypeOfFileValue is not a File🚫 Cursor Required

🚫 Null Checks
Operator JavaScript Key Description IndexedDB Optimized?
x == nullIsNullValue is null or undefined🚫 Cursor Required
x != nullIsNotNullValue is not null or undefined🚫 Cursor Required

📌 Query Additions (Sorting & Pagination)
LINQ Operation JavaScript Key Description
.OrderBy()orderBySort ascending
.OrderByDescending()orderByDescendingSort descending
.FirstOrDefaultAsync()firstGet first item
.LastOrDefaultAsync()lastGet last item
.Skip(x)skipSkip x results
.Take(x)takeTake x results
.TakeLast(x)takeLastTake last x results

⚠ IndexedDB has a reversed order for .Take() and .Skip()!
Always write .Take() before .Skip() for correct execution.


3. How Queries Are Structured

To integrate any language or framework, you need to convert expressions into the universal query format.

Example of a Universal Query Structure
{
  "nodeType": "logical",
  "operator": "And",
  "children": [
    {
      "nodeType": "condition",
      "condition": {
        "property": "age",
        "operation": "GreaterThan",
        "value": 30,
        "isString": false,
        "caseSensitive": false
      }
    },
    {
      "nodeType": "logical",
      "operator": "Or",
      "children": [
        {
          "nodeType": "condition",
          "condition": {
            "property": "city",
            "operation": "Equal",
            "value": "New York"
          }
        },
        {
          "nodeType": "condition",
          "condition": {
            "property": "city",
            "operation": "Equal",
            "value": "San Francisco"
          }
        }
      ]
    }
  ]
}

This query is equivalent to:

await personQuery
    .Where(x => x.Age > 30 && (x.City == "New York" || x.City == "San Francisco"))
    .ToListAsync();

🔍 Overview: Universal Predicate Language (UPL)

Purpose:
This format allows any programming language to serialize complex, nested logical filters (AND/OR) into a universal, portable structure that can be interpreted by any system (like your Magic IndexedDB engine, SQL, NoSQL, etc.).


🌲 Visual Map Example (for the JSON you provided)
AND
├── CONDITION: age > 30
└── OR
    ├── CONDITION: city == "New York"
    └── CONDITION: city == "San Francisco"

This shows:

📌 A top-level AND

📌 The left child is a condition (age > 30)

📌 The right child is a nested OR with two conditions


🧱 JSON Structure Explained
type PredicateNode =
  | LogicalNode
  | ConditionNode;

interface LogicalNode {
  nodeType: "logical";
  operator: "And" | "Or";
  children: PredicateNode[];
}

interface ConditionNode {
  nodeType: "condition",
  condition: {
    property: string,
    operation: "Equal" | "NotEqual" | "GreaterThan" | "LessThan" | ...,
    value: any,
    isString?: boolean,
    caseSensitive?: boolean
  }
}

This schema lets you nest as deeply as needed, mixing ANDs and ORs, enabling arbitrarily complex logic trees.


🛠️ Use Case: Translating Native Predicates

Here’s how to implement this in your language:

C# (LINQ)
x => x.Age > 30 && (x.City == "New York" || x.City == "San Francisco")

Becomes:

{
  "nodeType": "logical",
  "operator": "And",
  "children": [
    {
      "nodeType": "condition",
      "condition": {
        "property": "Age",
        "operation": "GreaterThan",
        "value": 30
      }
    },
    {
      "nodeType": "logical",
      "operator": "Or",
      "children": [
        {
          "nodeType": "condition",
          "condition": {
            "property": "City",
            "operation": "Equal",
            "value": "New York"
          }
        },
        {
          "nodeType": "condition",
          "condition": {
            "property": "City",
            "operation": "Equal",
            "value": "San Francisco"
          }
        }
      ]
    }
  ]
}

📘 Reference: Supported Operations
Operation Meaning
Equalx == value
NotEqualx != value
GreaterThanx > value
LessThanx < value
Containsx contains value (string)
Inx in [value1, value2]
StartsWithx starts with value
EndsWithx ends with value

There's a lot more operations than this built in. Documentation of the universal translation layer is still being updated.


🧠 Developer Notes

You can nest any combination of AND and ORs.

Each logical node can have any number of children (not just 2).

Supports easy translation to/from:

— LINQ expressions

— SQL WHERE clauses

— JavaScript Array.filter chains

— MongoDB query documents

You can build a universal query editor UI using this structure.

The universal translation layer handles complex nested (&&) and (||) operations for you with flattening, optimizations, and more. This is how Magic IndexedDB can perform complex nested operations while IndexedDB by default cannot.


4. IndexedDB Store Structure (DB Schema & Configuration)

IMPORTANT NOTE:
This section for the structure of the DB Schema is highly likely to change until the migration code is completely released. The future migration implementation will require adjustments here.

Before executing queries, Magic IndexedDB must understand how your database is structured. The system defines database stores dynamically based on schemas, ensuring proper indexing, compound keys, and validation for optimized queries.

📌 Example of a DB Store Schema
{
  "name": "MyDatabase",
  "version": 1,
  "storeSchemas": [
    {
      "tableName": "Users",
      "primaryKeyAuto": false,
      "columnNamesInCompoundKey": ["UserId", "TenantId"],
      "uniqueIndexes": ["Email"],
      "indexes": ["FirstName", "LastName", "CreatedAt"],
      "columnNamesInCompoundIndex": [[ "LastName", "FirstName" ]]
    },
    {
      "tableName": "Orders",
      "primaryKeyAuto": true,
      "columnNamesInCompoundKey": ["OrderId"],
      "uniqueIndexes": [],
      "indexes": ["UserId", "Status"],
      "columnNamesInCompoundIndex": [[ "UserId", "Status" ]]
    }
  ]
}
🔍 Breakdown of Each Property
Property Type Description
namestringName of the database
versionnumberDatabase version (used for migrations)
storeSchemasarrayList of table definitions
storeSchemas[].tableNamestringName of the table (IndexedDB object store)
storeSchemas[].primaryKeyAutobooleanWhether the primary key auto-increments
storeSchemas[].columnNamesInCompoundKeyarray<string>Columns forming a compound primary key
storeSchemas[].uniqueIndexesarray<string>Columns with unique constraints
storeSchemas[].indexesarray<string>Indexable columns for optimized queries
storeSchemas[].columnNamesInCompoundIndexarray<array<string>>Compound indexes for faster lookups

🚀 Why This Matters:

The schema is critical for ensuring that indexed queries work properly.

Unique indexes prevent duplicate values in specific fields (e.g., Email).

Compound indexes allow multi-column optimizations for advanced filtering.

If your database schema is not properly structured, queries may default to slower cursor-based scans instead of optimized index lookups.


5. Executing Queries

Once validated, queries are executed using Dexie.js and Magic IndexedDB’s optimized universal translation layer.

Yield-Based Query Execution
export async function* magicQueryYield(dbName, storeName, nestedOrFilter, queryAdditions = [], forceCursor = false) {
    if (!isValidFilterObject(nestedOrFilter)) {
        throw new Error("Invalid filter object provided.");
    }
    if (!isValidQueryAdditions(queryAdditions)) {
        throw new Error("Invalid query additions.");
    }

    // Execute Dexie.js query
}

🔹 Queries use yield to return results as they arrive.

🔹 A non-yield version (magicQueryAsync) is also available.


6. Integrating Your Own Framework

To integrate a new language or framework, follow these steps:

Step 1: Convert Expressions into Universal Query Format

🔄 Parse your language’s LINQ or query expressions.

🧩 Map them to QUERY_OPERATIONS and QUERY_ADDITIONS.

Step 2: Send Queries to the Universal Layer

⚙ Use magicQueryAsync() or magicQueryYield() to execute queries.

Step 3: Handle Results

⏳ Use async iteration (for await) for streaming data.

📦 Use bulk fetching (magicQueryAsync()) for standard queries.


7. Database & Table Handling

This process is still under development.


8. Contributing to the Universal Layer

If you're building a wrapper for a new framework:

🔧 Fork the repository.

📬 Submit a PR with your integration.

🌍 Join the community!

🔥 The dream? A universal LINQ to IndexedDB library for every programming language!

Let's make LINQ to IndexedDB a thing for all languages. Let’s make a better internet together.


9. Summary

Magic IndexedDB is a universal LINQ to IndexedDB system.

Supports multiple languages & frameworks through its universal query format.

Validation & translation ensure queries are safe & optimized.

Dexie.js powers efficient IndexedDB interactions.

Integrate your framework with a simple expression parser!

An unhandled error has occurred. Reload 🗙