Delete Data

Use the TypeScript client to delete data from a Synnax cluster.

The Synnax client allows deletion of time ranges of data in any channel: after each deletion operation is complete, all future reads will no longer include the deleted data. However, it may take a while before the underlying file sizes decrease – this allows deletion operations to be served in a rapid manner and only actually collect the unwanted data when the load on the server is low.

Note the differences between deleting data and deleting a channel – once a channel is deleted, it no longer exists; whereas when some data in a channel is deleted, we can write over that time range with new data or even delete some more data. Even if an entire channel’s data is deleted, the channel is still in the database, albeit empty.

Deleting Data From a Channel

The delete method of the client allows deletion of data (not to be confused with the delete method of the Channel class, which deletes channels). To delete a chunk of data, simply pass in the channel name(s) or key(s) and the time range to delete. As throughout Synnax, remember that a time range is start-inclusive and end-exclusive, i.e. data at the start time stamp is deleted and data at the end time stamp is not.

For example, to remove data in the range [00:01, 00:03) on the timestamps and my_precise_tc channels:

import { Synnax } from "@synnaxlabs/client"
import { TimeStamp } from "./telem";
const client = new Synnax(...)

// timestamps and my_precise_tc are two channels containing data.
await client.delete(
    ["my_index_timestamps", "my_precise_tc"],
    TimeStamp.seconds(1).range(TimeStamp.seconds(3))
)

Using channel name(s) to delete data will delete data in all channels with the given name(s). Using keys to delete is more preferable to prevent accidental deletion!

Note that delete is idempotent, meaning consecutive calls to delete on overlapping time ranges are allowed:

// no additional data deleted
await client.delete(
  ["my_index_timestamps", "my_precise_tc"],
  TimeStamp.seconds(1).range(TimeStamp.seconds(3)),
);

// 00:01 to 00:10 deleted
await client.delete(
  ["my_index_timestamps", "my_precise_tc"],
  TimeStamp.seconds(1).range(TimeStamp.seconds(10)),
);

Limitations of Deletions

In some situations, delete returns an error. If some channel keys or names do not exist in the database, the entirety of the delete operation fails, no data is deleted, and a NotFound error is returned:

// Suppose 111 and 112 are keys to channels that do exist, since 113 does not
// exist, none of these channels' data get deleted.
await client.delete([111, 112, 113], time_range_to_delete);

In the case where a requested channel is not found, delete is atomic: no data will be deleted and the operation will fail. However, in all other cases, delete is not atomic: failure in deleting data one channel halts the entire operation and returns an error immediately.

In addition, if a delete call is made to an index channel that other channels depend on data in the requested time range, an error is returned:

// If my_precise_tc is indexed by my_index_timestamps from 1 second to
// 3 seconds, we cannot delete my_index_timestamps. This call returns an error.
await client.delete(
  ["my_index_timestamps"],
  TimeStamp.seconds(1).range(TimeStamp.seconds(3)),
);

// If we delete my_precise_tc, the dependent, at the same time as
// my_index_timestamps, no errors are returned.
await client.delete(
  ["my_precise_tc", "my_index_timestamps"],
  TimeStamp.seconds(1).range(TimeStamp.seconds(3)),
);

Last but not least, delete calls on any channel with a writer whose start time is before the deleting time range return an error. This is to ensure that the writer and the deleter do not contend over data in the same region.

const writer = await client.openWriter({
  start: TimeStamp.seconds(10),
  channels: ["my_precise_tc"],
});

// error returned since writer start 00:10 is before deleting
// time range [00:12 - 00:30)
await client.delete(
  ["my_precise_tc"],
  TimeStamp.seconds(12).range(TimeStamp.seconds(30)),
);

Once writers starting before the deleting time range are closed, calls to delete may proceed normally.