2021-05-13 18:57:37 +02:00
|
|
|
import { knex, Knex } from 'knex';
|
2020-12-28 13:48:47 +02:00
|
|
|
import { DatabaseConfig } from './utils/types';
|
|
|
|
import * as pathUtils from 'path';
|
|
|
|
import time from '@joplin/lib/time';
|
|
|
|
import Logger from '@joplin/lib/Logger';
|
|
|
|
|
|
|
|
// Make sure bigInteger values are numbers and not strings
|
|
|
|
//
|
|
|
|
// https://github.com/brianc/node-pg-types
|
|
|
|
//
|
|
|
|
// In our case, all bigInteger are timestamps, which JavaScript can handle
|
|
|
|
// fine as numbers.
|
|
|
|
require('pg').types.setTypeParser(20, function(val: any) {
|
|
|
|
return parseInt(val, 10);
|
|
|
|
});
|
|
|
|
|
|
|
|
const logger = Logger.create('db');
|
|
|
|
|
|
|
|
const migrationDir = `${__dirname}/migrations`;
|
2021-05-25 12:13:35 +02:00
|
|
|
export const sqliteDefaultDir = pathUtils.dirname(__dirname);
|
2020-12-28 13:48:47 +02:00
|
|
|
|
2020-12-30 20:35:18 +02:00
|
|
|
export const defaultAdminEmail = 'admin@localhost';
|
|
|
|
export const defaultAdminPassword = 'admin';
|
|
|
|
|
2020-12-28 13:48:47 +02:00
|
|
|
export type DbConnection = Knex;
|
|
|
|
|
|
|
|
export interface DbConfigConnection {
|
|
|
|
host?: string;
|
|
|
|
port?: number;
|
|
|
|
user?: string;
|
|
|
|
database?: string;
|
|
|
|
filename?: string;
|
|
|
|
password?: string;
|
|
|
|
}
|
|
|
|
|
|
|
|
export interface KnexDatabaseConfig {
|
|
|
|
client: string;
|
|
|
|
connection: DbConfigConnection;
|
|
|
|
useNullAsDefault?: boolean;
|
|
|
|
asyncStackTraces?: boolean;
|
|
|
|
}
|
|
|
|
|
|
|
|
export interface ConnectionCheckResult {
|
|
|
|
isCreated: boolean;
|
|
|
|
error: any;
|
|
|
|
latestMigration: any;
|
|
|
|
connection: DbConnection;
|
|
|
|
}
|
|
|
|
|
|
|
|
export function makeKnexConfig(dbConfig: DatabaseConfig): KnexDatabaseConfig {
|
|
|
|
const connection: DbConfigConnection = {};
|
|
|
|
|
|
|
|
if (dbConfig.client === 'sqlite3') {
|
2021-05-25 12:13:35 +02:00
|
|
|
connection.filename = dbConfig.name;
|
2020-12-28 13:48:47 +02:00
|
|
|
} else {
|
|
|
|
connection.database = dbConfig.name;
|
|
|
|
connection.host = dbConfig.host;
|
|
|
|
connection.port = dbConfig.port;
|
|
|
|
connection.user = dbConfig.user;
|
|
|
|
connection.password = dbConfig.password;
|
|
|
|
}
|
|
|
|
|
|
|
|
return {
|
|
|
|
client: dbConfig.client,
|
|
|
|
useNullAsDefault: dbConfig.client === 'sqlite3',
|
|
|
|
asyncStackTraces: dbConfig.asyncStackTraces,
|
|
|
|
connection,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
export async function waitForConnection(dbConfig: DatabaseConfig): Promise<ConnectionCheckResult> {
|
|
|
|
const timeout = 30000;
|
|
|
|
const startTime = Date.now();
|
|
|
|
let lastError = { message: '' };
|
|
|
|
|
|
|
|
while (true) {
|
|
|
|
try {
|
|
|
|
const connection = await connectDb(dbConfig);
|
|
|
|
const check = await connectionCheck(connection);
|
|
|
|
if (check.error) throw check.error;
|
|
|
|
return check;
|
|
|
|
} catch (error) {
|
|
|
|
logger.info('Could not connect. Will try again.', error.message);
|
|
|
|
lastError = error;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Date.now() - startTime > timeout) {
|
|
|
|
logger.error('Timeout trying to connect to database:', lastError);
|
|
|
|
throw new Error(`Timeout trying to connect to database. Last error was: ${lastError.message}`);
|
|
|
|
}
|
|
|
|
|
|
|
|
await time.msleep(1000);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export async function connectDb(dbConfig: DatabaseConfig): Promise<DbConnection> {
|
2021-06-17 17:51:25 +02:00
|
|
|
const connection = knex(makeKnexConfig(dbConfig));
|
|
|
|
|
|
|
|
const debugSlowQueries = false;
|
|
|
|
|
|
|
|
if (debugSlowQueries) {
|
|
|
|
const startTimes: Record<string, number> = {};
|
|
|
|
|
|
|
|
const slowQueryDuration = 10;
|
|
|
|
|
|
|
|
connection.on('query', (data) => {
|
|
|
|
startTimes[data.__knexQueryUid] = Date.now();
|
|
|
|
});
|
|
|
|
|
|
|
|
connection.on('query-response', (_response, data) => {
|
|
|
|
const duration = Date.now() - startTimes[data.__knexQueryUid];
|
|
|
|
if (duration < slowQueryDuration) return;
|
|
|
|
console.info(`SQL: ${data.sql} (${duration}ms)`);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
return connection;
|
2020-12-28 13:48:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
export async function disconnectDb(db: DbConnection) {
|
|
|
|
await db.destroy();
|
|
|
|
}
|
|
|
|
|
2021-08-14 18:49:01 +02:00
|
|
|
export async function migrateLatest(db: DbConnection) {
|
2020-12-28 13:48:47 +02:00
|
|
|
await db.migrate.latest({
|
|
|
|
directory: migrationDir,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2021-08-14 18:49:01 +02:00
|
|
|
export async function migrateUp(db: DbConnection) {
|
|
|
|
await db.migrate.up({
|
|
|
|
directory: migrationDir,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
export async function migrateDown(db: DbConnection) {
|
|
|
|
await db.migrate.down({
|
|
|
|
directory: migrationDir,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
export async function migrateList(db: DbConnection, asString: boolean = true) {
|
|
|
|
const migrations: any = await db.migrate.list({
|
|
|
|
directory: migrationDir,
|
|
|
|
});
|
|
|
|
|
|
|
|
// The migration array has a rather inconsistent format:
|
|
|
|
//
|
|
|
|
// [
|
|
|
|
// // Done migrations
|
|
|
|
// [
|
|
|
|
// '20210809222118_email_key_fix.js',
|
|
|
|
// '20210814123815_testing.js',
|
|
|
|
// '20210814123816_testing.js'
|
|
|
|
// ],
|
|
|
|
// // Not done migrations
|
|
|
|
// [
|
|
|
|
// {
|
|
|
|
// file: '20210814123817_testing.js',
|
|
|
|
// directory: '/path/to/packages/server/dist/migrations'
|
|
|
|
// }
|
|
|
|
// ]
|
|
|
|
// ]
|
|
|
|
|
|
|
|
if (!asString) return migrations;
|
|
|
|
|
|
|
|
const formatName = (migrationInfo: any) => {
|
|
|
|
const name = migrationInfo.file ? migrationInfo.file : migrationInfo;
|
|
|
|
|
|
|
|
const s = name.split('.');
|
|
|
|
s.pop();
|
|
|
|
return s.join('.');
|
|
|
|
};
|
|
|
|
|
|
|
|
interface Line {
|
|
|
|
text: string;
|
|
|
|
done: boolean;
|
|
|
|
}
|
|
|
|
|
|
|
|
const output: Line[] = [];
|
|
|
|
|
|
|
|
for (const s of migrations[0]) {
|
|
|
|
output.push({
|
|
|
|
text: formatName(s),
|
|
|
|
done: true,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
for (const s of migrations[1]) {
|
|
|
|
output.push({
|
|
|
|
text: formatName(s),
|
|
|
|
done: false,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
output.sort((a, b) => {
|
|
|
|
return a.text < b.text ? -1 : +1;
|
|
|
|
});
|
|
|
|
|
|
|
|
return output.map(l => `${l.done ? '✓' : '✗'} ${l.text}`).join('\n');
|
|
|
|
}
|
|
|
|
|
2020-12-28 13:48:47 +02:00
|
|
|
function allTableNames(): string[] {
|
|
|
|
const tableNames = Object.keys(databaseSchema);
|
|
|
|
tableNames.push('knex_migrations');
|
|
|
|
tableNames.push('knex_migrations_lock');
|
|
|
|
return tableNames;
|
|
|
|
}
|
|
|
|
|
|
|
|
export async function dropTables(db: DbConnection): Promise<void> {
|
|
|
|
for (const tableName of allTableNames()) {
|
|
|
|
try {
|
|
|
|
await db.schema.dropTable(tableName);
|
|
|
|
} catch (error) {
|
|
|
|
if (isNoSuchTableError(error)) continue;
|
|
|
|
throw error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-12-30 20:35:18 +02:00
|
|
|
export async function truncateTables(db: DbConnection): Promise<void> {
|
|
|
|
for (const tableName of allTableNames()) {
|
|
|
|
try {
|
|
|
|
await db(tableName).truncate();
|
|
|
|
} catch (error) {
|
|
|
|
if (isNoSuchTableError(error)) continue;
|
|
|
|
throw error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-12-28 13:48:47 +02:00
|
|
|
function isNoSuchTableError(error: any): boolean {
|
|
|
|
if (error) {
|
|
|
|
// Postgres error: 42P01: undefined_table
|
|
|
|
if (error.code === '42P01') return true;
|
|
|
|
|
|
|
|
// Sqlite3 error
|
2021-05-13 18:57:37 +02:00
|
|
|
if (error.message && error.message.includes('SQLITE_ERROR: no such table:')) return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
export function isUniqueConstraintError(error: any): boolean {
|
|
|
|
if (error) {
|
|
|
|
// Postgres error: 23505: unique_violation
|
|
|
|
if (error.code === '23505') return true;
|
|
|
|
|
|
|
|
// Sqlite3 error
|
|
|
|
if (error.code === 'SQLITE_CONSTRAINT' && error.message.includes('UNIQUE constraint')) return true;
|
2020-12-28 13:48:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
export async function latestMigration(db: DbConnection): Promise<any> {
|
|
|
|
try {
|
|
|
|
const result = await db('knex_migrations').select('name').orderBy('id', 'asc').first();
|
|
|
|
return result;
|
|
|
|
} catch (error) {
|
|
|
|
// If the database has never been initialized, we return null, so
|
|
|
|
// for this we need to check the error code, which will be
|
|
|
|
// different depending on the DBMS.
|
|
|
|
|
|
|
|
if (isNoSuchTableError(error)) return null;
|
|
|
|
|
|
|
|
throw error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export async function connectionCheck(db: DbConnection): Promise<ConnectionCheckResult> {
|
|
|
|
try {
|
|
|
|
const result = await latestMigration(db);
|
|
|
|
return {
|
|
|
|
latestMigration: result,
|
|
|
|
isCreated: !!result,
|
|
|
|
error: null,
|
|
|
|
connection: db,
|
|
|
|
};
|
|
|
|
} catch (error) {
|
|
|
|
return {
|
|
|
|
latestMigration: null,
|
|
|
|
isCreated: false,
|
|
|
|
error: error,
|
|
|
|
connection: null,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export type Uuid = string;
|
|
|
|
|
|
|
|
export enum ItemAddressingType {
|
|
|
|
Id = 1,
|
|
|
|
Path,
|
|
|
|
}
|
|
|
|
|
2020-12-30 20:35:18 +02:00
|
|
|
export enum NotificationLevel {
|
|
|
|
Important = 10,
|
|
|
|
Normal = 20,
|
|
|
|
}
|
|
|
|
|
2020-12-28 13:48:47 +02:00
|
|
|
export enum ItemType {
|
2021-05-13 18:57:37 +02:00
|
|
|
Item = 1,
|
|
|
|
UserItem = 2,
|
|
|
|
User,
|
2020-12-28 13:48:47 +02:00
|
|
|
}
|
|
|
|
|
2021-05-25 11:49:47 +02:00
|
|
|
export enum EmailSender {
|
|
|
|
NoReply = 1,
|
|
|
|
Support = 2,
|
|
|
|
}
|
|
|
|
|
2020-12-28 13:48:47 +02:00
|
|
|
export enum ChangeType {
|
|
|
|
Create = 1,
|
|
|
|
Update = 2,
|
|
|
|
Delete = 3,
|
|
|
|
}
|
|
|
|
|
2021-05-13 18:57:37 +02:00
|
|
|
export enum FileContentType {
|
|
|
|
Any = 1,
|
|
|
|
JoplinItem = 2,
|
|
|
|
}
|
|
|
|
|
2021-03-20 19:09:55 +02:00
|
|
|
export function changeTypeToString(t: ChangeType): string {
|
|
|
|
if (t === ChangeType.Create) return 'create';
|
|
|
|
if (t === ChangeType.Update) return 'update';
|
|
|
|
if (t === ChangeType.Delete) return 'delete';
|
|
|
|
throw new Error(`Unkown type: ${t}`);
|
|
|
|
}
|
|
|
|
|
2021-01-29 20:45:11 +02:00
|
|
|
export enum ShareType {
|
2021-05-16 12:46:58 +02:00
|
|
|
Note = 1, // When a note is shared via a public link
|
|
|
|
Folder = 3, // When a complete folder is shared with another Joplin Server user
|
2021-05-13 18:57:37 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
export enum ShareUserStatus {
|
|
|
|
Waiting = 0,
|
|
|
|
Accepted = 1,
|
|
|
|
Rejected = 2,
|
2021-01-29 20:45:11 +02:00
|
|
|
}
|
|
|
|
|
2020-12-28 13:48:47 +02:00
|
|
|
export interface WithDates {
|
|
|
|
updated_time?: number;
|
|
|
|
created_time?: number;
|
|
|
|
}
|
|
|
|
|
|
|
|
export interface WithUuid {
|
2021-05-13 18:57:37 +02:00
|
|
|
id?: Uuid;
|
2020-12-28 13:48:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
interface DatabaseTableColumn {
|
|
|
|
type: string;
|
|
|
|
}
|
|
|
|
|
|
|
|
interface DatabaseTable {
|
|
|
|
[key: string]: DatabaseTableColumn;
|
|
|
|
}
|
|
|
|
|
|
|
|
interface DatabaseTables {
|
|
|
|
[key: string]: DatabaseTable;
|
|
|
|
}
|
|
|
|
|
|
|
|
// AUTO-GENERATED-TYPES
|
|
|
|
// Auto-generated using `npm run generate-types`
|
|
|
|
export interface Session extends WithDates, WithUuid {
|
|
|
|
user_id?: Uuid;
|
|
|
|
auth_code?: string;
|
|
|
|
}
|
|
|
|
|
2021-05-13 18:57:37 +02:00
|
|
|
export interface File {
|
|
|
|
id?: Uuid;
|
2020-12-28 13:48:47 +02:00
|
|
|
owner_id?: Uuid;
|
|
|
|
name?: string;
|
2021-05-13 18:57:37 +02:00
|
|
|
content?: any;
|
2020-12-28 13:48:47 +02:00
|
|
|
mime_type?: string;
|
|
|
|
size?: number;
|
|
|
|
is_directory?: number;
|
|
|
|
is_root?: number;
|
|
|
|
parent_id?: Uuid;
|
2021-05-13 18:57:37 +02:00
|
|
|
updated_time?: string;
|
|
|
|
created_time?: string;
|
|
|
|
source_file_id?: Uuid;
|
|
|
|
content_type?: number;
|
|
|
|
content_id?: Uuid;
|
2020-12-28 13:48:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
export interface ApiClient extends WithDates, WithUuid {
|
|
|
|
name?: string;
|
|
|
|
secret?: string;
|
|
|
|
}
|
|
|
|
|
2020-12-30 20:35:18 +02:00
|
|
|
export interface Notification extends WithDates, WithUuid {
|
|
|
|
owner_id?: Uuid;
|
|
|
|
level?: NotificationLevel;
|
|
|
|
key?: string;
|
|
|
|
message?: string;
|
|
|
|
read?: number;
|
|
|
|
canBeDismissed?: number;
|
|
|
|
}
|
|
|
|
|
2021-05-13 18:57:37 +02:00
|
|
|
export interface ShareUser extends WithDates, WithUuid {
|
|
|
|
share_id?: Uuid;
|
|
|
|
user_id?: Uuid;
|
|
|
|
status?: ShareUserStatus;
|
|
|
|
}
|
|
|
|
|
|
|
|
export interface Item extends WithDates, WithUuid {
|
|
|
|
name?: string;
|
|
|
|
mime_type?: string;
|
|
|
|
content?: Buffer;
|
|
|
|
content_size?: number;
|
|
|
|
jop_id?: Uuid;
|
|
|
|
jop_parent_id?: Uuid;
|
|
|
|
jop_share_id?: Uuid;
|
|
|
|
jop_type?: number;
|
|
|
|
jop_encryption_applied?: number;
|
2021-06-21 22:31:40 +02:00
|
|
|
jop_updated_time?: number;
|
2021-05-13 18:57:37 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
export interface UserItem extends WithDates {
|
|
|
|
id?: number;
|
|
|
|
user_id?: Uuid;
|
|
|
|
item_id?: Uuid;
|
|
|
|
}
|
|
|
|
|
|
|
|
export interface ItemResource {
|
|
|
|
id?: number;
|
|
|
|
item_id?: Uuid;
|
|
|
|
resource_id?: Uuid;
|
|
|
|
}
|
|
|
|
|
|
|
|
export interface KeyValue {
|
|
|
|
id?: number;
|
|
|
|
key?: string;
|
|
|
|
type?: number;
|
|
|
|
value?: string;
|
|
|
|
}
|
|
|
|
|
2021-01-29 20:45:11 +02:00
|
|
|
export interface Share extends WithDates, WithUuid {
|
|
|
|
owner_id?: Uuid;
|
2021-05-13 18:57:37 +02:00
|
|
|
item_id?: Uuid;
|
2021-01-29 20:45:11 +02:00
|
|
|
type?: ShareType;
|
2021-05-13 18:57:37 +02:00
|
|
|
folder_id?: Uuid;
|
|
|
|
note_id?: Uuid;
|
|
|
|
}
|
|
|
|
|
|
|
|
export interface Change extends WithDates, WithUuid {
|
|
|
|
counter?: number;
|
|
|
|
item_type?: ItemType;
|
|
|
|
item_id?: Uuid;
|
|
|
|
item_name?: string;
|
|
|
|
type?: ChangeType;
|
|
|
|
previous_item?: string;
|
|
|
|
user_id?: Uuid;
|
2021-01-29 20:45:11 +02:00
|
|
|
}
|
|
|
|
|
2021-05-25 11:49:47 +02:00
|
|
|
export interface Email extends WithDates {
|
|
|
|
id?: number;
|
|
|
|
recipient_name?: string;
|
|
|
|
recipient_email?: string;
|
|
|
|
recipient_id?: Uuid;
|
|
|
|
sender_id?: EmailSender;
|
|
|
|
subject?: string;
|
|
|
|
body?: string;
|
|
|
|
sent_time?: number;
|
|
|
|
sent_success?: number;
|
|
|
|
error?: string;
|
2021-08-09 17:55:04 +02:00
|
|
|
key?: string;
|
2021-05-25 11:49:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
export interface Token extends WithDates {
|
|
|
|
id?: number;
|
|
|
|
value?: string;
|
|
|
|
user_id?: Uuid;
|
|
|
|
}
|
|
|
|
|
2021-06-03 15:21:02 +02:00
|
|
|
export interface Subscription {
|
|
|
|
id?: number;
|
|
|
|
user_id?: Uuid;
|
|
|
|
stripe_user_id?: Uuid;
|
|
|
|
stripe_subscription_id?: Uuid;
|
|
|
|
last_payment_time?: number;
|
|
|
|
last_payment_failed_time?: number;
|
|
|
|
updated_time?: string;
|
|
|
|
created_time?: string;
|
2021-07-22 18:32:10 +02:00
|
|
|
is_deleted?: number;
|
2021-06-03 15:21:02 +02:00
|
|
|
}
|
|
|
|
|
2021-07-03 16:27:55 +02:00
|
|
|
export interface User extends WithDates, WithUuid {
|
|
|
|
email?: string;
|
|
|
|
password?: string;
|
|
|
|
full_name?: string;
|
|
|
|
is_admin?: number;
|
|
|
|
email_confirmed?: number;
|
|
|
|
must_set_password?: number;
|
|
|
|
account_type?: number;
|
|
|
|
can_upload?: number;
|
|
|
|
max_item_size?: number | null;
|
|
|
|
can_share_folder?: number | null;
|
|
|
|
can_share_note?: number | null;
|
|
|
|
max_total_item_size?: number | null;
|
|
|
|
total_item_size?: number;
|
2021-07-22 18:32:10 +02:00
|
|
|
enabled?: number;
|
2021-07-03 16:27:55 +02:00
|
|
|
}
|
|
|
|
|
2020-12-28 13:48:47 +02:00
|
|
|
export const databaseSchema: DatabaseTables = {
|
|
|
|
sessions: {
|
|
|
|
id: { type: 'string' },
|
|
|
|
user_id: { type: 'string' },
|
|
|
|
auth_code: { type: 'string' },
|
|
|
|
updated_time: { type: 'string' },
|
|
|
|
created_time: { type: 'string' },
|
|
|
|
},
|
|
|
|
files: {
|
|
|
|
id: { type: 'string' },
|
|
|
|
owner_id: { type: 'string' },
|
|
|
|
name: { type: 'string' },
|
|
|
|
content: { type: 'any' },
|
|
|
|
mime_type: { type: 'string' },
|
|
|
|
size: { type: 'number' },
|
|
|
|
is_directory: { type: 'number' },
|
|
|
|
is_root: { type: 'number' },
|
|
|
|
parent_id: { type: 'string' },
|
|
|
|
updated_time: { type: 'string' },
|
|
|
|
created_time: { type: 'string' },
|
2021-05-13 18:57:37 +02:00
|
|
|
source_file_id: { type: 'string' },
|
|
|
|
content_type: { type: 'number' },
|
|
|
|
content_id: { type: 'string' },
|
2020-12-28 13:48:47 +02:00
|
|
|
},
|
|
|
|
api_clients: {
|
|
|
|
id: { type: 'string' },
|
|
|
|
name: { type: 'string' },
|
|
|
|
secret: { type: 'string' },
|
|
|
|
updated_time: { type: 'string' },
|
|
|
|
created_time: { type: 'string' },
|
|
|
|
},
|
2020-12-30 20:35:18 +02:00
|
|
|
notifications: {
|
|
|
|
id: { type: 'string' },
|
|
|
|
owner_id: { type: 'string' },
|
|
|
|
level: { type: 'number' },
|
|
|
|
key: { type: 'string' },
|
|
|
|
message: { type: 'string' },
|
|
|
|
read: { type: 'number' },
|
|
|
|
canBeDismissed: { type: 'number' },
|
|
|
|
updated_time: { type: 'string' },
|
|
|
|
created_time: { type: 'string' },
|
|
|
|
},
|
2021-05-13 18:57:37 +02:00
|
|
|
share_users: {
|
|
|
|
id: { type: 'string' },
|
|
|
|
share_id: { type: 'string' },
|
|
|
|
user_id: { type: 'string' },
|
|
|
|
status: { type: 'number' },
|
|
|
|
updated_time: { type: 'string' },
|
|
|
|
created_time: { type: 'string' },
|
|
|
|
},
|
|
|
|
items: {
|
|
|
|
id: { type: 'string' },
|
|
|
|
name: { type: 'string' },
|
|
|
|
mime_type: { type: 'string' },
|
|
|
|
updated_time: { type: 'string' },
|
|
|
|
created_time: { type: 'string' },
|
|
|
|
content: { type: 'any' },
|
|
|
|
content_size: { type: 'number' },
|
|
|
|
jop_id: { type: 'string' },
|
|
|
|
jop_parent_id: { type: 'string' },
|
|
|
|
jop_share_id: { type: 'string' },
|
|
|
|
jop_type: { type: 'number' },
|
|
|
|
jop_encryption_applied: { type: 'number' },
|
2021-06-21 20:06:44 +02:00
|
|
|
jop_updated_time: { type: 'string' },
|
2021-05-13 18:57:37 +02:00
|
|
|
},
|
|
|
|
user_items: {
|
|
|
|
id: { type: 'number' },
|
|
|
|
user_id: { type: 'string' },
|
|
|
|
item_id: { type: 'string' },
|
|
|
|
updated_time: { type: 'string' },
|
|
|
|
created_time: { type: 'string' },
|
|
|
|
},
|
|
|
|
item_resources: {
|
|
|
|
id: { type: 'number' },
|
|
|
|
item_id: { type: 'string' },
|
|
|
|
resource_id: { type: 'string' },
|
|
|
|
},
|
|
|
|
key_values: {
|
|
|
|
id: { type: 'number' },
|
|
|
|
key: { type: 'string' },
|
|
|
|
type: { type: 'number' },
|
|
|
|
value: { type: 'string' },
|
|
|
|
},
|
2021-01-29 20:45:11 +02:00
|
|
|
shares: {
|
|
|
|
id: { type: 'string' },
|
|
|
|
owner_id: { type: 'string' },
|
2021-05-13 18:57:37 +02:00
|
|
|
item_id: { type: 'string' },
|
2021-01-29 20:45:11 +02:00
|
|
|
type: { type: 'number' },
|
|
|
|
updated_time: { type: 'string' },
|
|
|
|
created_time: { type: 'string' },
|
2021-05-13 18:57:37 +02:00
|
|
|
folder_id: { type: 'string' },
|
|
|
|
note_id: { type: 'string' },
|
|
|
|
},
|
|
|
|
changes: {
|
|
|
|
counter: { type: 'number' },
|
|
|
|
id: { type: 'string' },
|
|
|
|
item_type: { type: 'number' },
|
|
|
|
item_id: { type: 'string' },
|
|
|
|
item_name: { type: 'string' },
|
|
|
|
type: { type: 'number' },
|
|
|
|
updated_time: { type: 'string' },
|
|
|
|
created_time: { type: 'string' },
|
|
|
|
previous_item: { type: 'string' },
|
|
|
|
user_id: { type: 'string' },
|
2021-01-29 20:45:11 +02:00
|
|
|
},
|
2021-05-25 11:49:47 +02:00
|
|
|
emails: {
|
|
|
|
id: { type: 'number' },
|
|
|
|
recipient_name: { type: 'string' },
|
|
|
|
recipient_email: { type: 'string' },
|
|
|
|
recipient_id: { type: 'string' },
|
|
|
|
sender_id: { type: 'number' },
|
|
|
|
subject: { type: 'string' },
|
|
|
|
body: { type: 'string' },
|
|
|
|
sent_time: { type: 'string' },
|
|
|
|
sent_success: { type: 'number' },
|
|
|
|
error: { type: 'string' },
|
|
|
|
updated_time: { type: 'string' },
|
|
|
|
created_time: { type: 'string' },
|
2021-08-09 17:55:04 +02:00
|
|
|
key: { type: 'string' },
|
2021-05-25 11:49:47 +02:00
|
|
|
},
|
|
|
|
tokens: {
|
|
|
|
id: { type: 'number' },
|
|
|
|
value: { type: 'string' },
|
|
|
|
user_id: { type: 'string' },
|
|
|
|
updated_time: { type: 'string' },
|
|
|
|
created_time: { type: 'string' },
|
|
|
|
},
|
2021-06-03 15:21:02 +02:00
|
|
|
subscriptions: {
|
|
|
|
id: { type: 'number' },
|
|
|
|
user_id: { type: 'string' },
|
|
|
|
stripe_user_id: { type: 'string' },
|
|
|
|
stripe_subscription_id: { type: 'string' },
|
|
|
|
last_payment_time: { type: 'string' },
|
|
|
|
last_payment_failed_time: { type: 'string' },
|
|
|
|
updated_time: { type: 'string' },
|
|
|
|
created_time: { type: 'string' },
|
2021-07-22 18:32:10 +02:00
|
|
|
is_deleted: { type: 'number' },
|
2021-06-03 15:21:02 +02:00
|
|
|
},
|
2021-07-03 16:27:55 +02:00
|
|
|
users: {
|
|
|
|
id: { type: 'string' },
|
|
|
|
email: { type: 'string' },
|
|
|
|
password: { type: 'string' },
|
|
|
|
full_name: { type: 'string' },
|
|
|
|
is_admin: { type: 'number' },
|
|
|
|
updated_time: { type: 'string' },
|
|
|
|
created_time: { type: 'string' },
|
|
|
|
email_confirmed: { type: 'number' },
|
|
|
|
must_set_password: { type: 'number' },
|
|
|
|
account_type: { type: 'number' },
|
|
|
|
can_upload: { type: 'number' },
|
|
|
|
max_item_size: { type: 'number' },
|
|
|
|
can_share_folder: { type: 'number' },
|
|
|
|
can_share_note: { type: 'number' },
|
|
|
|
max_total_item_size: { type: 'string' },
|
2021-07-03 21:37:27 +02:00
|
|
|
total_item_size: { type: 'string' },
|
2021-07-22 18:32:10 +02:00
|
|
|
enabled: { type: 'number' },
|
2021-07-03 16:27:55 +02:00
|
|
|
},
|
2020-12-28 13:48:47 +02:00
|
|
|
};
|
|
|
|
// AUTO-GENERATED-TYPES
|