Skip to content

Latest commit

 

History

History
740 lines (664 loc) · 28.9 KB

ImportExportJson.md

File metadata and controls

740 lines (664 loc) · 28.9 KB


JSON Import/Export DOCUMENTATION

@capacitor-community/sqlite

Capacitor community plugin for Native and Electron SQLite Databases. For Native, databases could be encrypted with SQLCipher

Index

Methods

All the methods below give you all the bits and pieces to manage in your application the synchronization of SQL databases between a remote server and the mobile device. It can also be used for upgrading the schema of databases by exporting the current database to json, make the schema modifications in the json object and importing it back with the mode "full".

🚨 Release 3.4.3-2 Web, iOS & Android only ->> 🚨

The main change is related to the delete table's rows when a synchronization table exists as well as a last_mofidied table's column, allowing for database synchronization of the local database with a remote server database.

  • All existing triggers to YOUR_TABLE_NAME_trigger_last_modified must be modified as follows

    CREATE TRIGGER YOUR_TABLE_NAME_trigger_last_modified
      AFTER UPDATE ON YOUR_TABLE_NAME
      FOR EACH ROW WHEN NEW.last_modified <= OLD.last_modified
      BEGIN
          UPDATE YOUR_TABLE_NAME SET last_modified= (strftime('%s', 'now')) WHERE id=NEW.id;
      END;
    
  • an new column sql_deleted must be added to each of your tables as

    sql_deleted BOOLEAN DEFAULT 0 CHECK (sql_deleted IN (0, 1))
    

    This column will be autommatically set to 1 when you will use a DELETE FROM ... sql statement in the execute, run or executeSet methods.

  • In the JSON object that you provide to importFromJson, all the deleted rows in your remote server database's tables must have the sql_deleted column set to 1. This will indicate to the import process to physically delete the corresponding rows in your local database. All the others rows must have the sql_deleted column set to 0.

  • In the JSON object outputs by the exportToJson, all the deleted rows in your local database have got the sql_deleted column set to 1 to help in your synchronization management process with the remote server database. A system last_exported_date is automatically saved in the synchronization table at the start of the export process flow.

  • On successfull completion of your synchronization management process with the remote server database, you must

    • Set a new synchronization date (as (new Date()).toISOString()) with the setSyncDate method.
    • Execute the deleteExportedRows method which physically deletes all table's rows having 1 as value for the sql_deleted column prior to the last_exported_date in your local database.

An example of using this new feature is given in solidjs-vite-sqlite-app. It has been used to test the validity of the implementation.

🚨 Release 3.4.3-2 <<- 🚨

importFromJson

This method allow to create a database from a JSON Object. The created database can be encrypted or not based on the value of the name encrypted" of the JSON Object.

Prior to ececute this method, the connection to the database must be closed.

The import mode can be selected either full or partial

To use the partial mode, it is mandatory to add a column last_modified to the schema of all tables in your database.

When a table schema is created if a last_modified column exists, a trigger YOURTABLENAME_trigger_last_modified is automatically added to make sure that the last_modified column is updated with the date of the modification to allow the synchronization process when exporting back the data to the server.

🚨 Since release 3.4.2 ->> 🚨

The trigger before 3.4.2 was

CREATE TRIGGER IF NOT EXISTS YOURTABLENAME_trigger_last_modified AFTER UPDATE ON YOURTABLENAME FOR EACH ROW WHEN NEW.last_modified <= OLD.last_modified BEGIN UPDATE YOURTABLENAME SET last_modified = (strftime('%s', 'now')) WHERE id=OLD.id; END;

and is now and onwards to avoid updating the last_modified when no other values have been updated.

YOU MUST DELETE IT AND RE-CREATE IT FOR EACH OF YOUR TABLES AS BELOW:

CREATE TRIGGER IF NOT EXISTS YOURTABLENAME_trigger_last_modified AFTER UPDATE ON YOURTABLENAME FOR EACH ROW WHEN NEW.last_modified < OLD.last_modified BEGIN UPDATE YOURTABLENAME SET last_modified = (strftime('%s', 'now')) WHERE id=OLD.id; END; 🚨 Since release 3.4.2 <<- 🚨

When no last_modified column added:

  • full and partial modes are available for import,
  • only full mode is available for export

When mode full is choosen, all the existing tables and views are dropped if the database exists and they already exists in that database.

When mode partial is choosen, you can perform the following actions on an existing database

  • create new tables with indexes and data,
  • create new indexes to existing tables,
  • inserting new data to existing tables,
  • updating existing data to existing tables.
  • create new views

When mode partial, if you include views and/or schema of tables already existing, the views and/or tables will not be modified.

Internally the importFromJsonmethod is splitted into three SQL Transactions:

  • transaction building the schema (Tables, Indexes)
  • transaction creating the Table's Data (Insert, Update)
  • transaction creating the Views

🚨 Since release 3.4.2-3 ->> 🚨

  • overwrite boolean parameter has been added to the Json Object (default false)

    • true : delete physically the database whatever the version is.
    • false:
      • re-importing a database with the same version number will do nothing, keeping the existing database and will return changes = 0
      • re-importing a database with a lower version number will throw an error ImportFromJson: Cannot import a version lower than
  • During an import in full mode the Foreign Key constraint has been turn off before dropping the tables and turn back on after

🚨 Since release 3.4.2-3 <<- 🚨

exportToJson

This method allow to download a database to a Json object.

The export mode can be selected either full or partial

To use the partial mode, it is mandatory to add a field last_modified to the schema of all tables in your database. The export to Json will take all the schema, indexes or data which have been modified after the synchronization date.

isJsonValid

this method allow to check if the Json Object is valid before processing an import or validating the resulting Json Object from an export.

createSyncTable

Should be use once to create the table where the synchronization date is stored. To create a synchronization table, a column last_modified must be present in the database table's schema.

setSyncDate

Allow for updating the synchronization date. Only available if the synchronization table has been created.

getSyncDate

Allow for retreiving the synchronization date. Only available if the synchronization table has been created.

JSON Object

The JSON object is built up using the following types

It is mandatory that the first column in your database schema for all the tables is a primary key. This is requested to identify if the value given is an INSERT or an UPDATE SQL command to be executed.

export type jsonSQLite = {
  database: string,
  version: number,
  overwrite: boolean,    // since 3.4.2-3
  encrypted: boolean,
  mode: string,
  tables: Array<jsonTable>,
  views?: Array<jsonView>,
};

export type jsonTable = {
  name: string,
  schema?: Array<jsonColumn>,
  indexes?: Array<jsonIndex>,
  values?: Array<Array<any>>,
};
!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
Modified in 2.9.7 to allow for CONSTRAINT PRIMARY KEY with combined columns
!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
export type jsonColumn = {
  column?: string,
  foreignkey?: string,
  constraint?: string,
  value: string,
};
!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
Modified in 2.9.2 to allow for INDEX UNIQUE and with combined columns
!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
export type jsonIndex = {
  name: string,
  value: string,
  mode?: string
};
!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
Modified in 2.9.8 to allow import/export triggers
!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
export type JsonTrigger = {
  name: string,
  timeevent: string,
  condition?: string,
  logic: string,
};
!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
Added in 3.2.2-1 to allow for View creation
!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
export type jsonView = {
  name: string,
  value: string
};

JSON Template Examples

Full Mode One Step

const dataToImport: jsonSQLite = {
  database: 'db-from-json',
  version: 1,
  encrypted: false,
  mode: 'full',
  tables: [
    {
      name: 'users',
      schema: [
        { column: 'id', value: 'INTEGER PRIMARY KEY NOT NULL' },
        { column: 'email', value: 'TEXT UNIQUE NOT NULL' },
        { column: 'name', value: 'TEXT' },
        { column: 'age', value: 'INTEGER' },
        {column:"sql_deleted", value:"BOOLEAN DEFAULT 0 CHECK (sql_deleted IN (0, 1))"},
        {column:"last_modified", value:"INTEGER DEFAULT (strftime('%s', 'now'))"}
      ],
      indexes: [
        { name: 'index_user_on_name', value: 'name' },
        { name: 'index_user_on_last_modified', value: 'last_modified DESC' },
        {
          name: 'index_user_on_email_name',
          value: 'email ASC, name',
          mode: 'UNIQUE',
        },
      ],
      values: [
        [1, 'Whiteley.com', 'Whiteley', 30, 0, 1582536810],
        [2, 'Jones.com', 'Jones', 44, 0, 1582812800],
        [3, 'Simpson@example.com', 'Simpson', 69, 0, 1583570630],
        [4, 'Brown@example.com', 'Brown', 15, 0, 1590383895],
      ],
    },
    {
      name: 'messages',
      schema: [
        { column: 'id', value: 'INTEGER PRIMARY KEY NOT NULL' },
        { column: 'userid', value: 'INTEGER' },
        { column: 'title', value: 'TEXT NOT NULL' },
        { column: 'body', value: 'TEXT NOT NULL' },
        {column:"sql_deleted", value:"BOOLEAN DEFAULT 0 CHECK (sql_deleted IN (0, 1))"},
        {column:"last_modified", value:"INTEGER DEFAULT (strftime('%s', 'now'))"},
        {
          foreignkey: 'userid',
          value: 'REFERENCES users(id) ON DELETE CASCADE',
        },
      ],
      indexes: [
        { name: 'index_messages_on_title', value: 'title' },
        {
          name: 'index_messages_on_last_modified',
          value: 'last_modified DESC',
        },
      ],
      values: [
        [1, 1, 'test post 1', 'content test post 1', 0, 1587310030],
        [2, 2, 'test post 2', 'content test post 2', 0, 1590388125],
        [3, 1, 'test post 3', 'content test post 3', 0, 1590383895],
      ],
    },
    {
      name: 'images',
      schema: [
        { column: 'id', value: 'INTEGER PRIMARY KEY NOT NULL' },
        { column: 'name', value: 'TEXT UNIQUE NOT NULL' },
        { column: 'type', value: 'TEXT NOT NULL' },
        { column: 'size', value: 'INTEGER' },
        { column: 'img', value: 'BLOB' },
        {column:"sql_deleted", value:"BOOLEAN DEFAULT 0 CHECK (sql_deleted IN (0, 1))"},
        {column:"last_modified", value:"INTEGER DEFAULT (strftime('%s', 'now'))"}
      ],
      indexes: [
        { name: 'index_images_on_last_modified', value: 'last_modified DESC' },
      ],
      values: [
        [1, 'meowth', 'png', 'NULL', Images[0], 0, 1590388825],
        [2, 'feather', 'png', 'NULL', Images[1], 0, 1590389895],
      ],
    },
  ],
};

Images are defined as base64 strings

const Images: Array<string> = [
    "",
    ""
];

Full Mode Two Steps

  • first the database schema
const dataToImport1: jsonSQLite = {
  database: 'db-from-json',
  version: 1,
  encrypted: false,
  mode: 'full',
  tables: [
    {
      name: 'users',
      schema: [
        { column: 'id', value: 'INTEGER PRIMARY KEY NOT NULL' },
        { column: 'email', value: 'TEXT UNIQUE NOT NULL' },
        { column: 'name', value: 'TEXT' },
        { column: 'age', value: 'INTEGER' },
        {column:"sql_deleted", value:"BOOLEAN DEFAULT 0 CHECK (sql_deleted IN (0, 1))"},
        {column:"last_modified", value:"INTEGER DEFAULT (strftime('%s', 'now'))"}
      ],
      indexes: [
        { name: 'index_user_on_name', value: 'name' },
        { name: 'index_user_on_last_modified', value: 'last_modified DESC' },
        {
          name: 'index_user_on_email_name',
          value: 'email ASC, name',
          mode: 'UNIQUE',
        },
      ],
    },
    {
      name: 'messages',
      schema: [
        { column: 'id', value: 'INTEGER PRIMARY KEY NOT NULL' },
        { column: 'userid', value: 'INTEGER' },
        { column: 'title', value: 'TEXT NOT NULL' },
        { column: 'body', value: 'TEXT NOT NULL' },
        {column:"sql_deleted", value:"BOOLEAN DEFAULT 0 CHECK (sql_deleted IN (0, 1))"},
        {column:"last_modified", value:"INTEGER DEFAULT (strftime('%s', 'now'))"},
        {
          foreignkey: 'userid',
          value: 'REFERENCES users(id) ON DELETE CASCADE',
        },
      ],
      indexes: [
        {
          name: 'index_messages_on_last_modified',
          value: 'last_modified DESC',
        },
      ],
    },
  ],
};
  • followed by an import of the Table' Data
const dataToImport2: jsonSQLite = {
  database: 'db-from-json',
  version: 1,
  encrypted: false,
  mode: 'full',
  tables: [
    {
      name: 'users',
      values: [
        [1, 'Whiteley.com', 'Whiteley', 30, 0, 1582536810],
        [2, 'Jones.com', 'Jones', 44, 0, 1582812800],
        [3, 'Simpson@example.com', 'Simpson', 69, 0, 1583570630],
        [4, 'Brown@example.com', 'Brown', 15, 0, 1590383895],
      ],
    },
    {
      name: 'messages',
      values: [
        [1, 1, 'test post 1', 'content test post 1', 0, 1587310030],
        [2, 2, 'test post 2', 'content test post 2', 0, 1590388125],
        [3, 1, 'test post 3', 'content test post 3', 0, 1590383895],
      ],
    },
  ],
};

Partial Mode

const partialImport1: any = {
    database : "db-from-json",
    version: 1,
    encrypted : false,
    mode : "partial",
    tables :[
        {
          name: "users",
          values: [
              [5,"Addington.com","Addington",22,0,1601972413],
              [6,"Bannister.com","Bannister",59,0,1601983245],
              [2,"Jones@example.com","Jones",45,0,1601995473],
              [1, 'Whiteley.com', 'Whiteley', 30, 1, 1601995520],
          ]
        },
        {
          name: "messages",
          indexes: [
              {name: "index_messages_on_title", value: "title"}
          ],
          values: [
              [4,2,"test post 4","content test post 4",0,1601983742],
              [5,6,"test post 5","content test post 5",0,1601992872]
              [1,1,'test post 1', 'content test post 1',1,1601995520],
              [3,1,'test post 3', 'content test post 3',1,1601995520],
         ]
        },
        {
          name: 'fruits',
          schema: [
            { column: 'id', value: 'INTEGER PRIMARY KEY NOT NULL' },
            { column: 'name', value: 'TEXT UNIQUE NOT NULL' },
            { column: 'weight', value: 'REAL' },
            {column:"sql_deleted", value:"BOOLEAN DEFAULT 0 CHECK (sql_deleted IN (0, 1))"},
            {column:"last_modified", value:"INTEGER DEFAULT (strftime('%s', 'now'))"}
          ],
          indexes: [
            { name: 'index_fruits_on_name', value: 'name' },
            { name: "index_fruits_on_last_modified",value: "last_modified DESC"},
          ],
          values: [
            [1, 'orange', 200.3,0,1601995573],
            [2, 'apple', 450.0,0,1601995573],
            [2, 'banana', 120.5,0,1601995573],
          ],
        },
        {
          name: 'company',
          schema: [
            { column: 'id', value: 'INTEGER NOT NULL' },
            { column: 'name', value: 'TEXT NOT NULL' },
            { column: 'age', value: 'INTEGER NOT NULL' },
            { column: 'address', value: 'TEXT' },
            { column: 'salary', value: 'REAL'},
            {column:"sql_deleted", value:"BOOLEAN DEFAULT 0 CHECK (sql_deleted IN (0, 1))"},
            {column:"last_modified", value:"INTEGER DEFAULT (strftime('%s', 'now'))"},
            { constraint: 'PK_id_name', value: 'PRIMARY KEY (id,name)'},
          ],
        },
    ],
};

Full Mode with Triggers and UUID

export const dataToImport59: any = {
  database: 'db-from-json59',
  version: 1,
  encrypted: false,
  mode: 'full',
  tables: [
    {
      name: 'countries',
      schema: [
        { column: 'id', value: 'TEXT PRIMARY KEY NOT NULL' },
        { column: 'name', value: 'TEXT UNIQUE NOT NULL' },
        { column: 'code', value: 'TEXT' },
        { column: 'language', value: 'TEXT' },
        { column: 'phone_code', value: 'TEXT' },
        {column:"sql_deleted", value:"BOOLEAN DEFAULT 0 CHECK (sql_deleted IN (0, 1))"},
        {column:"last_modified", value:"INTEGER DEFAULT (strftime('%s', 'now'))"}
      ],
      indexes: [
        { name: 'index_country_on_name', value: 'name', mode: 'UNIQUE' },
        { name: 'index_country_on_last_modified', value: 'last_modified DESC' },
      ],
      values: [
        ['3', 'Afghanistan', 'AF', 'fa', '93', 0, 1608216034],
        ['6', 'Albania', 'AL', 'sq', '355', 0, 1608216034],
        ['56', 'Algeria', 'DZ', 'ar', '213', 0, 1608216034],
      ],
    },
    {
      name: 'customers',
      schema: [
        { column: 'id', value: 'TEXT PRIMARY KEY NOT NULL' },
        { column: 'first_name', value: 'TEXT NOT NULL' },
        { column: 'last_name', value: 'TEXT NOT NULL' },
        { column: 'gender', value: 'TEXT NOT NULL' },
        { column: 'email', value: 'TEXT UNIQUE NOT NULL' },
        { column: 'phone', value: 'TEXT' },
        { column: 'national_id', value: 'TEXT NOT NULL' },
        { column: 'date_of_birth', value: 'TEXT' },
        { column: 'created_at', value: 'TEXT' },
        { column: 'created_by', value: 'TEXT' },
        { column: 'last_edited', value: 'TEXT' },
        { column: 'organization', value: 'TEXT' },
        { column: 'comment_id', value: 'TEXT' },
        { column: 'country_id', value: 'TEXT NOT NULL' },
        {column:"sql_deleted", value:"BOOLEAN DEFAULT 0 CHECK (sql_deleted IN (0, 1))"},
        {column:"last_modified", value:"INTEGER DEFAULT (strftime('%s', 'now'))"},
        {
          foreignkey: 'country_id',
          value: 'REFERENCES countries(id) ON DELETE CASCADE',
        },
      ],
      indexes: [
        { name: 'index_customers_on_email', value: 'email', mode: 'UNIQUE' },
        {
          name: 'index_customers_on_last_modified',
          value: 'last_modified DESC',
        },
      ],
      triggers: [
        {
          name: 'validate_email_before_insert_customers',
          timeevent: 'BEFORE INSERT',
          logic:
            "BEGIN SELECT CASE WHEN NEW.email NOT LIKE '%_@__%.__%' THEN RAISE (ABORT,'Invalid email address') END; END",
        },
      ],
      values: [
        [
          'ef5c57d5-b885-49a9-9c4d-8b340e4abdbc',
          'William',
          'Jones',
          '1',
          'peterjones@mail.com<peterjones@mail.com>',
          '420305202',
          '1234567',
          '1983-01-04',
          '2020-11-1212:39:02',
          '3',
          '2020-11-19 05:10:10',
          '1',
          'NULL',
          '3',
          0,
          1608216040,
        ],
        [
          'bced3262-5d42-470a-9585-d3fd12c45452',
          'Alexander',
          'Brown',
          '1',
          'alexanderbrown@mail.com<alexanderbrown@mail.com>',
          '420305203',
          '1234572',
          '1990-02-07',
          '2020-12-1210:35:15',
          '1',
          '2020-11-19 05:10:10',
          '2',
          'NULL',
          '6',
          0,
          1608216040,
        ],
      ],
    },
  ],
};

Full Mode with Views

export const dataToImport167: any = {
  database: "db-issue167",
  version: 1,
  encrypted: false,
  mode: "full",
  tables: [
    {
      name: "departments",
      schema: [
        {column: "id", value: "INTEGER PRIMARY KEY AUTOINCREMENT" },
        {column: "name", value: "TEXT NOT NULL" },
        {column:"sql_deleted", value:"BOOLEAN DEFAULT 0 CHECK (sql_deleted IN (0, 1))"},
        {column:"last_modified", value:"INTEGER DEFAULT (strftime('%s', 'now'))"}
      ],
      indexes: [
        {name: "index_departments_on_last_modified",value: "last_modified DESC"}
      ],
      values: [
        [1,"Admin",0,1608216034],
        [2,"Sales",0,1608216034],
        [3,"Quality Control",0,1608216034],
        [4,"Marketing",0,1608216034],
      ]
    },
    {
      name: "employees",
      schema: [
        {column: "id", value: "INTEGER PRIMARY KEY AUTOINCREMENT" },
        {column: "first_name", value: "TEXT" },
        {column: "last_name", value: "TEXT" },
        {column: "salary", value: "NUMERIC" },
        {column: "dept_id", value: "INTEGER" },
        {column:"sql_deleted", value:"BOOLEAN DEFAULT 0 CHECK (sql_deleted IN (0, 1))"},
        {column:"last_modified", value:"INTEGER DEFAULT (strftime('%s', 'now'))"}
      ],
      indexes: [
        {name: "index_departments_on_last_modified",value: "last_modified DESC"}
      ],
      values: [
        [1,"John","Brown",27500,1,0,1608216034],
        [2,"Sally","Brown",37500,2,0,1608216034],
        [3,'Vinay','Jariwala', 35100,3,0,1608216034],
        [4,'Jagruti','Viras', 9500,2,0,1608216034],
        [5,'Shweta','Rana',12000,3,0,1608216034],
        [6,'sonal','Menpara', 13000,1,0,1608216034],
        [7,'Yamini','Patel', 10000,2,0,1608216034],
        [8,'Khyati','Shah', 50000,3,0,1608216034],
        [9,'Shwets','Jariwala',19400,2,0,1608216034],
        [10,'Kirk','Douglas',36400,4,0,1608216034],
        [11,'Leo','White',45000,4,0,1608216034],
      ],
    }
  ],
  views: [
    {name: "SalesTeam", value: "SELECT id,first_name,last_name from employees WHERE dept_id IN (SELECT id FROM departments where name='Sales')"},
    {name: "AdminTeam", value: "SELECT id,first_name,last_name from employees WHERE dept_id IN (SELECT id FROM departments where name='Admin')"},
  ]
}

Partial Mode with Views

export const viewsToImport167: any = {
  database: "db-issue167",
  version: 1,
  encrypted: false,
  mode: "partial",
  tables: [],
  views: [
    {name: "QualityControlTeam", value: "SELECT id,first_name,last_name from employees WHERE dept_id IN (SELECT id FROM departments where name='Quality Control')"},
    {name: "MarketingTeam", value: "SELECT id,first_name,last_name from employees WHERE dept_id IN (SELECT id FROM departments where name='Marketing')"},
  ]
}

Full Mode with Composite Foreign Key Constraints

export const schemaToImport179 = {
    database: 'db-issue179',
    version: 1,
    encrypted: false,
    mode: 'full',
    tables: [
      {
        name: 'album',
        schema: [
          { column: 'albumartist', value: 'TEXT NOT NULL' },
          { column: 'albumname', value: 'TEXT NOT NULL' },
          { column: 'albumcover', value: 'BINARY' },
          {column:"sql_deleted", value:"BOOLEAN DEFAULT 0 CHECK (sql_deleted IN (0, 1))"},
          {column:"last_modified", value:"INTEGER DEFAULT (strftime('%s', 'now'))"}
          { constraint: 'PK_albumartist_albumname', value: 'PRIMARY KEY (albumartist,albumname)'},
        ],
        indexes: [
          { name: 'index_album_on_albumartist_albumname', value: 'albumartist,albumname' },
          { name: 'index_album_on_last_modified', value: 'last_modified DESC' },
        ],
      },
      {
        name: 'song',
        schema: [
          { column: 'songid', value: 'INTEGER PRIMARY KEY NOT NULL' },
          { column: 'songartist', value: 'TEXT NOT NULL' },
          { column: 'songalbum', value: 'TEXT NOT NULL' },
          { column: 'songname', value: 'TEXT NOT NULL' },
          {column:"sql_deleted", value:"BOOLEAN DEFAULT 0 CHECK (sql_deleted IN (0, 1))"},
          {column:"last_modified", value:"INTEGER DEFAULT (strftime('%s', 'now'))"},
          {
            foreignkey: 'songartist,songalbum',
            value: 'REFERENCES album(albumartist,albumname)',
          },
        ],
        indexes: [
            { name: 'index_song_on_songartist_songalbum', value: 'songartist,songalbum' },
            {
            name: 'index_song_on_last_modified',
            value: 'last_modified DESC',
          },
        ],
      },
    ],
  };

Import with Affinity Names

const dataToImportFull71: any = {
  database : 'db-from-json71',
  version : 1,
  encrypted : false,
  mode : 'full',
  tables :[
      {
          name: 'company',
          schema: [
              {column:'id', value: 'INTEGER'},
              {column:'name', value:'VARCHAR(25) NOT NULL'},
              {column:'age', value:'INT NOT NULL'},
              {column:'country', value:'CHARACTER(20)'},
              {column:'salary', value:'DECIMAL(10,3)'},
              {column:'manager', value:'BOOLEAN DEFAULT 0 CHECK (manager IN (0, 1))'},
              {column:"sql_deleted", value:"BOOLEAN DEFAULT 0 CHECK (sql_deleted IN (0, 1))"},
              {column:"last_modified", value:"INTEGER DEFAULT (strftime('%s', 'now'))"},
              {constraint:'PK_name_country', value:'PRIMARY KEY (name,country)'}
          ],
          values: [
              [1,'Jones',55,'Australia',1250,1,0,1608216034],
              [2,'Lawson',32,'Ireland',2345.60,0,0,1608216034],
              [3,'Bush',44,'USA',1850.10,0,0,1608216034],
          ]
      },
  ]
};