SurrealDB
SurrealDB Docs Logo

Enter a search query

Navigation

equals()

The equals() function performs deep equality comparison for values, including SurrealDB-specific types that may not compare correctly with JavaScript’s === operator.

Import:

import { equals } from 'surrealdb';

Source: utils/equals.ts

Function Signature

function equals(a: unknown, b: unknown): boolean

Parameters

ParameterTypeDescription
a requiredunknownFirst value to compare.
b requiredunknownSecond value to compare.

Returns

boolean - true if values are deeply equal, false otherwise

What It Compares

The equals() function correctly handles:

  • Primitives - strings, numbers, booleans, null, undefined
  • SurrealDB Types - RecordId, DateTime, Duration, Decimal, Uuid, etc.
  • Dates - JavaScript Date objects
  • Regular Expressions - RegExp patterns
  • Objects - Deep object comparison
  • Arrays - Deep array comparison
  • Mixed types - Proper handling of bigint/number comparisons

Examples

Primitive Comparisons

import { equals } from 'surrealdb'; console.log(equals(42, 42)); // true console.log(equals('hello', 'hello')); // true console.log(equals(true, false)); // false console.log(equals(null, null)); // true

SurrealDB Type Comparisons

import { RecordId, DateTime, Uuid, Decimal, equals } from 'surrealdb'; // RecordId comparison const id1 = new RecordId('users', 'john'); const id2 = new RecordId('users', 'john'); const id3 = new RecordId('users', 'jane'); console.log(equals(id1, id2)); // true console.log(equals(id1, id3)); // false // DateTime comparison (including nanoseconds) const dt1 = DateTime.parse('2024-01-15T12:00:00.123456789Z'); const dt2 = DateTime.parse('2024-01-15T12:00:00.123456789Z'); console.log(equals(dt1, dt2)); // true // Uuid comparison const uuid1 = Uuid.parse('550e8400-e29b-41d4-a716-446655440000'); const uuid2 = Uuid.parse('550e8400-e29b-41d4-a716-446655440000'); console.log(equals(uuid1, uuid2)); // true // Decimal comparison (arbitrary precision) const dec1 = new Decimal('19.99'); const dec2 = new Decimal('19.99'); console.log(equals(dec1, dec2)); // true

Object Comparisons

const obj1 = { name: 'John', age: 30 }; const obj2 = { name: 'John', age: 30 }; const obj3 = { name: 'John', age: 31 }; console.log(equals(obj1, obj2)); // true console.log(equals(obj1, obj3)); // false // Deep nested objects const deep1 = { user: { profile: { name: 'John' } } }; const deep2 = { user: { profile: { name: 'John' } } }; console.log(equals(deep1, deep2)); // true

Array Comparisons

const arr1 = [1, 2, 3]; const arr2 = [1, 2, 3]; const arr3 = [1, 2, 4]; console.log(equals(arr1, arr2)); // true console.log(equals(arr1, arr3)); // false // Arrays with SurrealDB types const ids1 = [ new RecordId('users', 'john'), new RecordId('users', 'jane') ]; const ids2 = [ new RecordId('users', 'john'), new RecordId('users', 'jane') ]; console.log(equals(ids1, ids2)); // true

Mixed Type Comparisons

// bigint and number comparison console.log(equals(42n, 42)); // true console.log(equals(42, 42n)); // true // Date comparison const date1 = new Date('2024-01-15'); const date2 = new Date('2024-01-15'); console.log(equals(date1, date2)); // true

Use Cases

Checking Record Existence

async function hasRecord(recordId: RecordId): Promise<boolean> { const records = await db.select(new Table(recordId.tb)); return records.some(record => equals(record.id, recordId)); } const exists = await hasRecord(new RecordId('users', 'john'));

Deduplication

function deduplicate<T>(items: T[]): T[] { const unique: T[] = []; for (const item of items) { if (!unique.some(u => equals(u, item))) { unique.push(item); } } return unique; } const recordIds = [ new RecordId('users', 'john'), new RecordId('users', 'jane'), new RecordId('users', 'john'), // Duplicate ]; const uniqueIds = deduplicate(recordIds); console.log(uniqueIds.length); // 2

Change Detection

function hasChanged<T>(before: T, after: T): boolean { return !equals(before, after); } const originalUser = await db.select(userId); // ... user makes changes ... const updatedUser = await db.select(userId); if (hasChanged(originalUser, updatedUser)) { console.log('User data was modified'); }

Array Difference

function findNewItems<T>(oldList: T[], newList: T[]): T[] { return newList.filter(newItem => !oldList.some(oldItem => equals(oldItem, newItem)) ); } const oldTags = ['javascript', 'typescript']; const newTags = ['javascript', 'typescript', 'react']; const added = findNewItems(oldTags, newTags); console.log(added); // ['react']

Caching / Memoization

class Cache<K, V> { private cache = new Map<string, { key: K; value: V }>(); set(key: K, value: V): void { const entry = { key, value }; this.cache.set(JSON.stringify(key), entry); } get(key: K): V | undefined { for (const entry of this.cache.values()) { if (equals(entry.key, key)) { return entry.value; } } return undefined; } } // Use with complex keys const cache = new Cache<RecordId, User>(); const userId = new RecordId('users', 'john'); cache.set(userId, userData); const cached = cache.get(new RecordId('users', 'john')); // Found!

Record Comparison

interface User { id: RecordId; name: string; email: string; settings: { theme: string; notifications: boolean; }; } const user1: User = await db.select(new RecordId('users', 'john')); const user2: User = await db.select(new RecordId('users', 'john')); // Deep comparison including nested objects and RecordId console.log(equals(user1, user2)); // true

Why Not Use === or ==?

JavaScript’s equality operators don’t work correctly for:

// Problem 1: Object references const obj1 = { name: 'John' }; const obj2 = { name: 'John' }; console.log(obj1 === obj2); // false (different references) console.log(equals(obj1, obj2)); // true (same content) // Problem 2: SurrealDB types const id1 = new RecordId('users', 'john'); const id2 = new RecordId('users', 'john'); console.log(id1 === id2); // false (different instances) console.log(equals(id1, id2)); // true (same value) // Problem 3: Date comparison const date1 = new Date('2024-01-15'); const date2 = new Date('2024-01-15'); console.log(date1 === date2); // false console.log(equals(date1, date2)); // true // Problem 4: Arrays const arr1 = [1, 2, 3]; const arr2 = [1, 2, 3]; console.log(arr1 === arr2); // false console.log(equals(arr1, arr2)); // true

Best Practices

1. Use for Value Comparisons

// Good: Deep equality if (equals(recordId1, recordId2)) { // Same record } // Avoid: Reference equality (wrong for objects) if (recordId1 === recordId2) { // Only true if same instance }

2. Use for Complex Type Comparisons

// Good: Proper comparison const sameDateTime = equals( DateTime.parse('2024-01-15T12:00:00.123456789Z'), DateTime.parse('2024-01-15T12:00:00.123456789Z') ); // true // Avoid: Direct comparison const wrong = dt1 === dt2; // false (different instances)

3. Prefer Built-in .equals() for Single Types

const id1 = new RecordId('users', 'john'); const id2 = new RecordId('users', 'john'); // Both are fine: console.log(equals(id1, id2)); // true console.log(id1.equals(id2)); // true // Use id1.equals(id2) when you know both are RecordId // Use equals(id1, id2) for generic comparisons

See Also