1
0
mirror of https://github.com/laurent22/joplin.git synced 2024-12-30 10:36:35 +02:00
joplin/packages/app-cli/app/fuzzing.js

2403 lines
33 KiB
JavaScript

'use strict';
const time = require('@joplin/lib/time').default;
const Logger = require('@joplin/lib/Logger').default;
const Resource = require('@joplin/lib/models/Resource.js');
const { dirname } = require('@joplin/lib/path-utils');
const FsDriverNode = require('@joplin/lib/fs-driver-node').default;
const lodash = require('lodash');
const exec = require('child_process').exec;
const fs = require('fs-extra');
const baseDir = `${dirname(__dirname)}/tests/fuzzing`;
const syncDir = `${baseDir}/sync`;
const joplinAppPath = `${__dirname}/main.js`;
const syncDurations = [];
const fsDriver = new FsDriverNode();
Logger.fsDriver_ = fsDriver;
Resource.fsDriver_ = fsDriver;
const logger = new Logger();
logger.addTarget('console');
logger.setLevel(Logger.LEVEL_DEBUG);
process.on('unhandledRejection', (reason, p) => {
console.error('Unhandled promise rejection', p, 'reason:', reason);
});
function createClient(id) {
return {
id: id,
profileDir: `${baseDir}/client${id}`,
};
}
async function createClients() {
const output = [];
const promises = [];
for (let clientId = 0; clientId < 2; clientId++) {
const client = createClient(clientId);
promises.push(fs.remove(client.profileDir));
promises.push(
execCommand(client, 'config sync.target 2').then(() => {
return execCommand(client, `config sync.2.path ${syncDir}`);
})
);
output.push(client);
}
await Promise.all(promises);
return output;
}
function randomElement(array) {
if (!array.length) return null;
return array[Math.floor(Math.random() * array.length)];
}
function randomWord() {
const words = [
'belief',
'scandalous',
'flawless',
'wrestle',
'sort',
'moldy',
'carve',
'incompetent',
'cruel',
'awful',
'fang',
'holistic',
'makeshift',
'synonymous',
'questionable',
'soft',
'drop',
'boot',
'whimsical',
'stir',
'idea',
'adhesive',
'present',
'hilarious',
'unusual',
'divergent',
'probable',
'depend',
'suck',
'belong',
'advise',
'straight',
'encouraging',
'wing',
'clam',
'serve',
'fill',
'nostalgic',
'dysfunctional',
'aggressive',
'floor',
'baby',
'grease',
'sisters',
'print',
'switch',
'control',
'victorious',
'cracker',
'dream',
'wistful',
'adaptable',
'reminiscent',
'inquisitive',
'pushy',
'unaccountable',
'receive',
'guttural',
'two',
'protect',
'skin',
'unbiased',
'plastic',
'loutish',
'zip',
'used',
'divide',
'communicate',
'dear',
'muddled',
'dinosaurs',
'grip',
'trees',
'well-off',
'calendar',
'chickens',
'irate',
'deranged',
'trip',
'stream',
'white',
'poison',
'attack',
'obtain',
'theory',
'laborer',
'omniscient',
'brake',
'maniacal',
'curvy',
'smoke',
'babies',
'punch',
'hammer',
'toothbrush',
'same',
'crown',
'jagged',
'peep',
'difficult',
'reject',
'merciful',
'useless',
'doctor',
'mix',
'wicked',
'plant',
'quickest',
'roll',
'suffer',
'curly',
'brother',
'frighten',
'cold',
'tremendous',
'move',
'knot',
'lame',
'imaginary',
'capricious',
'raspy',
'aunt',
'loving',
'wink',
'wooden',
'hop',
'free',
'drab',
'fire',
'instrument',
'border',
'frame',
'silent',
'glue',
'decorate',
'distance',
'powerful',
'pig',
'admit',
'fix',
'pour',
'flesh',
'profuse',
'skinny',
'learn',
'filthy',
'dress',
'bloody',
'produce',
'innocent',
'meaty',
'pray',
'slimy',
'sun',
'kindhearted',
'dime',
'exclusive',
'boast',
'neat',
'ruthless',
'recess',
'grieving',
'daily',
'hateful',
'ignorant',
'fence',
'spring',
'slim',
'education',
'overflow',
'plastic',
'gaping',
'chew',
'detect',
'right',
'lunch',
'gainful',
'argue',
'cloistered',
'horses',
'orange',
'shame',
'bitter',
'able',
'sail',
'magical',
'exist',
'force',
'wheel',
'best',
'suit',
'spurious',
'partner',
'request',
'dog',
'gusty',
'money',
'gaze',
'lonely',
'company',
'pale',
'tempt',
'rat',
'flame',
'wobble',
'superficial',
'stop',
'protective',
'stare',
'tongue',
'heal',
'railway',
'idiotic',
'roll',
'puffy',
'turn',
'meeting',
'new',
'frightening',
'sophisticated',
'poke',
'elderly',
'room',
'stimulating',
'increase',
'moor',
'secret',
'lean',
'occur',
'country',
'damp',
'evanescent',
'alluring',
'oafish',
'join',
'thundering',
'cars',
'awesome',
'advice',
'unruly',
'ray',
'wind',
'anxious',
'fly',
'hammer',
'adventurous',
'shop',
'cook',
'trucks',
'nonchalant',
'addition',
'base',
'abashed',
'excuse',
'giants',
'dramatic',
'piquant',
'coach',
'possess',
'poor',
'finger',
'wide-eyed',
'aquatic',
'welcome',
'instruct',
'expert',
'evasive',
'hug',
'cute',
'return',
'mice',
'damage',
'turkey',
'quiet',
'bewildered',
'tidy',
'pointless',
'outrageous',
'medical',
'foolish',
'curve',
'grandiose',
'gullible',
'hapless',
'gleaming',
'third',
'grin',
'pipe',
'egg',
'act',
'physical',
'eager',
'side',
'milk',
'tearful',
'fertile',
'average',
'glamorous',
'strange',
'yak',
'terrific',
'thin',
'near',
'snails',
'flowery',
'authority',
'fish',
'curious',
'perpetual',
'healthy',
'health',
'match',
'fade',
'chemical',
'economic',
'drawer',
'avoid',
'lying',
'minister',
'lick',
'powder',
'decay',
'desire',
'furry',
'faint',
'beam',
'sordid',
'fax',
'tail',
'bawdy',
'cherry',
'letter',
'clover',
'ladybug',
'teeth',
'behavior',
'black',
'amazing',
'pink',
'waste',
'island',
'forgetful',
'needless',
'lock',
'waves',
'boundary',
'receipt',
'handy',
'religion',
'hypnotic',
'aftermath',
'explain',
'sense',
'mundane',
'rambunctious',
'second',
'preserve',
'alarm',
'dusty',
'event',
'blow',
'weigh',
'value',
'glorious',
'jail',
'sigh',
'cemetery',
'serious',
'yummy',
'cattle',
'understood',
'limit',
'alert',
'fear',
'lucky',
'tested',
'surround',
'dolls',
'pleasant',
'disillusioned',
'discover',
'tray',
'night',
'seemly',
'liquid',
'worry',
'pen',
'bent',
'gruesome',
'war',
'teeny-tiny',
'common',
'judge',
'symptomatic',
'bed',
'trot',
'unequaled',
'flowers',
'friends',
'damaged',
'peel',
'skip',
'show',
'twist',
'worthless',
'brush',
'look',
'behave',
'imperfect',
'week',
'petite',
'direction',
'soda',
'lively',
'coal',
'coil',
'release',
'berserk',
'books',
'impossible',
'replace',
'cough',
'chunky',
'torpid',
'discreet',
'material',
'bomb',
'soothe',
'crack',
'hope',
'license',
'frightened',
'breathe',
'maddening',
'calculator',
'committee',
'paltry',
'green',
'subsequent',
'arrest',
'gigantic',
'tasty',
'metal',
'willing',
'man',
'stem',
'nonstop',
'route',
'impulse',
'government',
'comfortable',
'include',
'literate',
'multiply',
'test',
'vast',
'exercise',
'addicted',
'agreeable',
'lace',
'toes',
'young',
'water',
'end',
'wash',
'glossy',
'round',
'staking',
'sink',
'open',
'spot',
'trip',
'fierce',
'robust',
'pastoral',
'drown',
'dress',
'machine',
'calculating',
'holiday',
'crabby',
'disgusting',
'plan',
'sleet',
'sleepy',
'typical',
'borrow',
'possible',
'curtain',
'airplane',
'industry',
'nut',
'rough',
'wacky',
'rock',
'enormous',
'uninterested',
'sugar',
'rake',
'consist',
'wrist',
'basket',
'chop',
'wet',
'street',
'known',
'settle',
'bless',
'cluttered',
'wild',
'expand',
'angle',
'snake',
'yawn',
'hate',
'flood',
'rabid',
'spiteful',
'anger',
'market',
'bizarre',
'force',
'majestic',
'scissors',
'beg',
'rifle',
'foregoing',
'cactus',
'funny',
'eggnog',
'wish',
'high-pitched',
'drop',
'camp',
'scarf',
'car',
'groan',
'wonderful',
'wealthy',
'cup',
'lock',
'available',
'previous',
'jam',
'political',
'vacation',
'three',
'desk',
'fry',
'aspiring',
'productive',
'clear',
'bored',
'flashy',
'plug',
'precede',
'abhorrent',
'muddle',
'flimsy',
'paste',
'need',
'reward',
'frail',
'obnoxious',
'creature',
'whip',
'unbecoming',
'lake',
'unused',
'chin',
'tour',
'zephyr',
'experience',
'building',
'scrub',
'correct',
'hover',
'panicky',
'scorch',
'diligent',
'hulking',
'ubiquitous',
'tedious',
'aberrant',
'file',
'accidental',
'mist',
'blue-eyed',
'trite',
'nondescript',
'cows',
'wait',
'test',
'snotty',
'amuck',
'jump',
'lackadaisical',
'grey',
'tawdry',
'strong',
'land',
'kind',
'star',
'ludicrous',
'stupid',
'telling',
'use',
'bruise',
'whirl',
'cream',
'harsh',
'aboriginal',
'substantial',
'brawny',
'tease',
'pollution',
'weather',
'degree',
'dry',
'film',
'obey',
'closed',
'dependent',
'want',
'undesirable',
'stamp',
'relax',
'foot',
'obscene',
'successful',
'wriggle',
'drain',
'greasy',
'escape',
'cross',
'odd',
'boring',
'absorbed',
'houses',
'suppose',
'suit',
'moon',
'ceaseless',
'explode',
'clap',
'pop',
'courageous',
'miss',
'notebook',
'delirious',
'form',
'pretty',
'sock',
'grotesque',
'noxious',
'record',
'stop',
'saw',
'thing',
'dislike',
'cloth',
'six',
'jar',
'unnatural',
'spiffy',
'itchy',
'secretary',
'move',
'certain',
'unkempt',
'sassy',
'queue',
'shrug',
'crow',
'heavenly',
'desert',
'screw',
'vessel',
'mug',
'encourage',
'icy',
'enthusiastic',
'throat',
'whistle',
'ignore',
'miniature',
'squeak',
'scarecrow',
'fluttering',
'hang',
'icicle',
'lie',
'juicy',
'empty',
'baseball',
'various',
'promise',
'abortive',
'descriptive',
'high',
'spy',
'faded',
'talk',
'air',
'messup',
'decorous',
'sneaky',
'mark',
'sack',
'ultra',
'chivalrous',
'lethal',
'expect',
'disgusted',
'reaction',
'fireman',
'private',
'ritzy',
'manage',
'actor',
'rely',
'uppity',
'thread',
'bat',
'space',
'underwear',
'blood',
'nine',
'maid',
'shelf',
'hanging',
'shop',
'prick',
'wound',
'sloppy',
'offer',
'increase',
'clear',
'slap',
'rude',
'poised',
'wretched',
'cause',
'quince',
'tame',
'remarkable',
'abject',
'sail',
'guide',
'subdued',
'spiky',
'debonair',
'chicken',
'tired',
'hum',
'land',
'scared',
'splendid',
'guess',
'cast',
'rub',
'magnificent',
'ants',
'overwrought',
'interfere',
'gorgeous',
'office',
'trade',
'sniff',
'melted',
'bore',
'point',
'pet',
'purple',
'brake',
'flavor',
'toe',
'prickly',
'zinc',
'homely',
'modern',
'kindly',
'whisper',
'bare',
'annoyed',
'glass',
'noisy',
'null',
'thoughtless',
'skirt',
'dock',
'rings',
'mind',
'neck',
'macho',
'wave',
'history',
'play',
'road',
'profit',
'word',
'opposite',
'dreary',
'governor',
'horse',
'trust',
'elbow',
'kiss',
'crayon',
'stitch',
'excited',
'needy',
'arrange',
'easy',
'alcoholic',
'safe',
'lumpy',
'monkey',
'smile',
'capable',
'untidy',
'extra-small',
'memory',
'selective',
'reproduce',
'old-fashioned',
'overrated',
'texture',
'knit',
'downtown',
'risk',
'pot',
'sofa',
'righteous',
'wren',
'pull',
'carry',
'aboard',
'listen',
'classy',
'thank',
'shocking',
'condition',
'root',
'attempt',
'swim',
'frog',
'hurt',
'army',
'title',
'handsomely',
'town',
'guiltless',
'thaw',
'spell',
'selfish',
'disturbed',
'tramp',
'girls',
'utopian',
'noiseless',
'trail',
'bashful',
'business',
'rhetorical',
'snail',
'sign',
'plausible',
'left',
'design',
'tall',
'violent',
'wasteful',
'beautiful',
'breezy',
'tap',
'murder',
'talented',
'needle',
'creator',
'imagine',
'flippant',
'dead',
'bone',
'coherent',
'relation',
'aromatic',
'mountainous',
'face',
'ask',
'picture',
'pedal',
'colour',
'obese',
'group',
'top',
'bubble',
'pinch',
'optimal',
'school',
'bathe',
'flagrant',
'check',
'deliver',
'pass',
'tan',
'crate',
'hose',
'debt',
'faulty',
'longing',
'hollow',
'invincible',
'afford',
'lovely',
'ticket',
'changeable',
'subtract',
'fumbling',
'responsible',
'confused',
'woman',
'touch',
'watch',
'zesty',
'library',
'jail',
'wrap',
'terrify',
'brick',
'popcorn',
'cooperative',
'peck',
'pocket',
'property',
'buzz',
'tiresome',
'digestion',
'exciting',
'nation',
'juvenile',
'shade',
'copper',
'wanting',
'deer',
'waste',
'man',
'join',
'spotty',
'amused',
'mountain',
'waggish',
'bushes',
'tense',
'river',
'heartbreaking',
'help',
'mine',
'narrow',
'smash',
'scrawny',
'tame',
'rain',
'playground',
'airport',
'astonishing',
'level',
'befitting',
'animal',
'heat',
'painful',
'cellar',
'ski',
'sedate',
'knowing',
'vigorous',
'change',
'eight',
'ship',
'work',
'strip',
'robin',
'tank',
'challenge',
'vacuous',
'representative',
'regret',
'tightfisted',
'erratic',
'club',
'imported',
'therapeutic',
'rainstorm',
'luxuriant',
'relieved',
'day',
'system',
'apologise',
'male',
'prepare',
'malicious',
'naive',
'whistle',
'curl',
'hobbies',
'trousers',
'stereotyped',
'dad',
'endurable',
'grass',
'hot',
'bomb',
'morning',
'guide',
'keen',
'plot',
'accept',
'disastrous',
'macabre',
'year',
'spicy',
'absorbing',
'sticks',
'efficient',
'drain',
'warm',
'rice',
'utter',
'fact',
'marked',
'ratty',
'chalk',
'towering',
'treat',
'nest',
'annoy',
'jealous',
'stamp',
'effect',
'cautious',
'jelly',
'feigned',
'gabby',
'corn',
'volleyball',
'pan',
'psychedelic',
'fairies',
'silent',
'zonked',
'bump',
'trouble',
'mass',
'queen',
'things',
'bury',
'sister',
'quiet',
'colossal',
'puncture',
'four',
'attend',
'love',
'wiry',
'vegetable',
'destruction',
'note',
'pies',
'resolute',
'load',
'fancy',
'tacky',
'periodic',
'abandoned',
'vivacious',
'blush',
'wrathful',
'miscreant',
'call',
'striped',
'wiggly',
'supreme',
'hand',
'impolite',
'rule',
'deserted',
'concern',
'cover',
'harbor',
'waiting',
'soggy',
'psychotic',
'ancient',
'sponge',
'domineering',
'elegant',
'impartial',
'unlock',
'abrasive',
'count',
'flight',
'neighborly',
'roof',
'bulb',
'auspicious',
'automatic',
'magic',
'sign',
'amusing',
'orange',
'branch',
'sulky',
'attack',
'fetch',
'number',
'jellyfish',
'start',
'alike',
'touch',
'sour',
'wary',
'minor',
'punish',
'connect',
'protest',
'pie',
'kaput',
'doubtful',
'friendly',
'simplistic',
'smart',
'vanish',
'applaud',
'jumbled',
'ready',
'yell',
'support',
'squash',
'raise',
'parallel',
'super',
'jazzy',
'crush',
'apathetic',
'water',
'food',
'thrill',
'permit',
'heady',
'last',
'mine',
'signal',
'smoke',
'preach',
'x-ray',
'name',
'birth',
'minute',
'steel',
'bedroom',
'female',
'acrid',
'riddle',
'attractive',
'earth',
'crack',
'muscle',
'alive',
'guarded',
'sweater',
'donkey',
'doubt',
'lettuce',
'magenta',
'live',
'farm',
'glib',
'bow',
'fascinated',
'friend',
'practise',
'remember',
'bleach',
'hungry',
'voiceless',
'pin',
'sparkling',
'report',
'arm',
'sad',
'shaggy',
'parcel',
'wail',
'flash',
'territory',
'functional',
'wise',
'screeching',
'appliance',
'future',
'appear',
'team',
'rabbit',
'porter',
'paint',
'flat',
'amusement',
'ocean',
'head',
'geese',
'wash',
'embarrassed',
'tub',
'boundless',
'freezing',
'mushy',
'surprise',
'temporary',
'marble',
'recondite',
'telephone',
'zipper',
'pine',
'reign',
'pump',
'tangy',
'reply',
'toys',
'purpose',
'songs',
'form',
'delicious',
'wood',
'horn',
'nutty',
'fruit',
'lumber',
'potato',
'cheat',
'cloudy',
'visit',
'reduce',
'destroy',
'deafening',
'full',
'warlike',
'mitten',
'cover',
'earthy',
'seashore',
'yarn',
'tenuous',
'pause',
'boil',
'dogs',
'tough',
'knife',
'shy',
'naughty',
'existence',
'fire',
'eminent',
'remove',
'juice',
'sleep',
'voyage',
'balance',
'unsightly',
'plough',
'ill-fated',
'pumped',
'motionless',
'allow',
'trade',
'warm',
'toad',
'wave',
'wall',
'pigs',
'circle',
'rejoice',
'ear',
'drink',
'found',
'taboo',
'object',
'old',
'temper',
'plant',
'public',
'picayune',
'blot',
'delight',
'carpenter',
'dispensable',
'tire',
'cow',
'furniture',
'rightful',
'mute',
'gentle',
'gifted',
'ragged',
'stiff',
'retire',
'compare',
'sable',
'hole',
'judicious',
'chilly',
'sparkle',
'futuristic',
'love',
'bubble',
'travel',
'name',
'numberless',
'succeed',
'acoustic',
'lowly',
'society',
'injure',
'agree',
'reason',
'party',
'wool',
'careful',
'hook',
'bell',
'ball',
'attach',
'scream',
'development',
'happy',
'appreciate',
'disagree',
'request',
'march',
'rampant',
'scrape',
'sack',
'hair',
'measure',
'owe',
'grubby',
'vein',
'boy',
'punishment',
'smoggy',
'wry',
'immense',
'shoe',
'pack',
'brash',
'cave',
'sincere',
'adorable',
'fantastic',
'attraction',
'racial',
'jittery',
'defiant',
'honey',
'paper',
'weight',
'bee',
'blind',
'birthday',
'toothsome',
'trick',
'guard',
'fog',
'handle',
'dirty',
'salt',
'rinse',
'nippy',
'observe',
'suggestion',
'weak',
'instinctive',
'frequent',
'detail',
'verse',
'quirky',
'scattered',
'toy',
'aware',
'distribution',
'repulsive',
'draconian',
'bucket',
'harm',
'radiate',
'bang',
'shrill',
'living',
'rhythm',
'obsequious',
'drum',
'inject',
'skate',
'beds',
'smash',
'order',
'stitch',
'ground',
'nosy',
'kick',
'dusty',
'home',
'rot',
'frame',
'jam',
'sky',
'soap',
'rescue',
'energetic',
'grape',
'massive',
'deeply',
'dazzling',
'park',
'pull',
'number',
'abundant',
'barbarous',
'drag',
'ajar',
'close',
'moan',
'haircut',
'shade',
'married',
'cats',
'thirsty',
'dirt',
'vagabond',
'fearful',
'squealing',
'squalid',
'zebra',
'murky',
'sheet',
'fat',
'follow',
'bikes',
'unpack',
'materialistic',
'surprise',
'arch',
'selection',
'acoustics',
'helpless',
'thoughtful',
'cry',
'quarrelsome',
'arrogant',
'illegal',
'sudden',
'elite',
'tomatoes',
'spoil',
'flower',
'shivering',
'front',
'caption',
'volcano',
'ugliest',
'ambitious',
'pickle',
'interrupt',
'nervous',
'approve',
'messy',
'dust',
'oceanic',
'brass',
'tremble',
'fine',
'nerve',
'lunchroom',
'hard',
'engine',
'erect',
'flower',
'cynical',
'irritating',
'tight',
'cobweb',
'gray',
'invention',
'snatch',
'account',
'sharp',
'spooky',
'squeamish',
'eatable',
'share',
'need',
'moaning',
'suspect',
'rush',
'rural',
'false',
'float',
'bite',
'careless',
'sidewalk',
'cowardly',
'stroke',
'educated',
'ugly',
'type',
'wandering',
'bolt',
'mint',
'fit',
'large',
'extra-large',
'defeated',
'kitty',
'tacit',
'abiding',
'grandfather',
'trains',
'lamp',
'habitual',
'fast',
'offbeat',
'accurate',
'many',
'fortunate',
'lyrical',
'charge',
'illustrious',
'transport',
'wakeful',
'cable',
'ordinary',
'string',
'question',
'train',
'fancy',
'kick',
'enchanting',
'jobless',
'ahead',
'comparison',
'loose',
'dance',
'add',
'wonder',
'stale',
'earn',
'reflective',
'bright',
'true',
'statuesque',
'amount',
'matter',
'repair',
'care',
'ruin',
'terrible',
'elastic',
'spiders',
'craven',
'lamentable',
'decision',
'swing',
'connection',
'gaudy',
'knowledge',
'cheap',
'lazy',
'step',
'dinner',
'rod',
'agreement',
'comb',
'mean',
'past',
'knotty',
'busy',
'quicksand',
'match',
'early',
'long',
'onerous',
'ambiguous',
'worried',
'spade',
'happen',
'crook',
'dapper',
'grate',
'announce',
'plate',
'haunt',
'friction',
'actually',
'chance',
'example',
'rapid',
'zealous',
'necessary',
'ink',
'mere',
'shock',
'huge',
'jaded',
'spill',
'store',
'fuzzy',
'table',
'bottle',
'halting',
'spark',
'end',
'remain',
'transport',
'seat',
'leg',
'long-term',
'clip',
'grumpy',
'shake',
'walk',
'try',
'action',
'soup',
'short',
'hurry',
'square',
'belligerent',
'thankful',
'beginner',
'small',
'bumpy',
'silly',
'badge',
'marvelous',
'wealth',
'open',
'unequal',
'scatter',
'pest',
'fool',
'step',
'groovy',
'childlike',
'door',
'bouncy',
'believe',
'incredible',
'box',
'unhealthy',
'swanky',
'abrupt',
'depressed',
'flaky',
'famous',
'detailed',
'regret',
'envious',
'natural',
'apparel',
'spare',
'mark',
'ten',
'power',
'glistening',
'arrive',
'animated',
'slip',
'heap',
'shaky',
'unfasten',
'contain',
'inexpensive',
'introduce',
'shallow',
'rule',
'gather',
'pump',
'humorous',
'acceptable',
'womanly',
'giddy',
'silk',
'yoke',
'straw',
'invite',
'one',
'red',
'growth',
'unadvised',
'measly',
'flap',
'puzzled',
'regular',
'painstaking',
'little',
'plain',
'tumble',
'rest',
'fabulous',
'melt',
'label',
'truculent',
'internal',
'passenger',
'zippy',
'bright',
'earsplitting',
'tooth',
'veil',
'grip',
'square',
'stuff',
'gate',
'level',
'stone',
'observation',
'time',
'workable',
'bird',
'realise',
'spotted',
'coast',
'quiver',
'rebel',
'entertain',
'rotten',
'loss',
'collect',
'meal',
'satisfy',
'military',
'bake',
'cagey',
'redundant',
'treatment',
'knock',
'blink',
'scale',
'board',
'fair',
'nebulous',
'sip',
'homeless',
'place',
'complain',
'joke',
'bat',
'winter',
'choke',
'frantic',
'chubby',
'highfalutin',
'troubled',
'whole',
'rose',
'delightful',
'loaf',
'afraid',
'sturdy',
'class',
'cultured',
'yielding',
'broken',
'kittens',
'absurd',
'discovery',
'next',
'disarm',
'dangerous',
'lively',
'reflect',
'chief',
'teeny',
'pencil',
'ban',
'grade',
'size',
'dashing',
'thought',
'breath',
'empty',
'hellish',
'shock',
'sea',
'weary',
'payment',
'limping',
'premium',
'grateful',
'somber',
'tax',
'coach',
'vulgar',
'stretch',
'tow',
'branch',
'insurance',
'yam',
'stormy',
'wish',
'snow',
'cute',
'milky',
'battle',
'far',
'roasted',
'slip',
'adamant',
'awake',
'employ',
'tangible',
'tickle',
'jog',
'hysterical',
'meddle',
'parsimonious',
'judge',
'educate',
'respect',
'sound',
'oven',
'gratis',
'station',
'train',
'purring',
'steady',
'carriage',
'humdrum',
'voracious',
'jolly',
'brainy',
'proud',
'elfin',
'cure',
'motion',
'record',
'quizzical',
'pail',
'bike',
'faithful',
'approval',
'vague',
'fall',
'store',
'normal',
'rock',
'bear',
'bounce',
'giant',
'satisfying',
'crooked',
'lopsided',
'vest',
'separate',
'sneeze',
'teaching',
'general',
'meat',
'festive',
'historical',
'line',
'north',
'tip',
'son',
'damaging',
'nimble',
'broad',
'list',
'confuse',
'first',
'deserve',
'steep',
'last',
'rich',
'oval',
'thick',
'glow',
'great',
'clammy',
'cheer',
'untidy',
'scientific',
'noise',
'stomach',
'undress',
'big',
'bite-sized',
'enter',
'cake',
'aloof',
'abnormal',
'month',
'grab',
'well-groomed',
'silver',
'art',
'berry',
'giraffe',
'complete',
'billowy',
'thumb',
'squeal',
'crib',
'discussion',
'even',
'stretch',
'mellow',
'angry',
'grouchy',
'absent',
'snow',
'middle',
'stingy',
'mourn',
'deep',
'honorable',
'nice',
'verdant',
'reach',
'lavish',
'sin',
'interest',
'whine',
'tug',
'vengeful',
'rhyme',
'stay',
'upset',
'hesitant',
'tent',
'wire',
'gold',
'momentous',
'yellow',
'cap',
'delicate',
'youthful',
'twig',
'burly',
'devilish',
'chess',
'wide',
'misty',
'useful',
'memorise',
'madly',
'plants',
'spectacular',
'accessible',
'collar',
'truck',
'harmony',
'uncovered',
'beef',
'low',
'channel',
'abusive',
'analyse',
'observant',
'snobbish',
'duck',
'excellent',
'intend',
'wreck',
'testy',
'care',
'shoes',
'charming',
'demonic',
'can',
'wipe',
'acidic',
'watch',
'decisive',
'brave',
'greet',
'imminent',
'influence',
'oranges',
'seal',
'eggs',
'knowledgeable',
'ashamed',
'shiny',
'inconclusive',
'remind',
'house',
'solid',
'quixotic',
'describe',
'support',
];
return randomElement(words);
}
function execCommand(client, command, options = {}) {
const exePath = `node ${joplinAppPath}`;
const cmd = `${exePath} --update-geolocation-disabled --env dev --log-level debug --profile ${client.profileDir} ${command}`;
logger.info(`${client.id}: ${command}`);
if (options.killAfter) {
logger.info(`Kill after: ${options.killAfter}`);
}
return new Promise((resolve, reject) => {
const childProcess = exec(cmd, (error, stdout, stderr) => {
if (error) {
if (error.signal == 'SIGTERM') {
resolve('Process was killed');
} else {
logger.error(stderr);
reject(error);
}
} else {
resolve(stdout.trim());
}
});
if (options.killAfter) {
setTimeout(() => {
logger.info('Sending kill signal...');
childProcess.kill();
}, options.killAfter);
}
});
}
async function clientItems(client) {
const itemsJson = await execCommand(client, 'dump');
try {
return JSON.parse(itemsJson);
} catch (error) {
throw new Error(`Cannot parse JSON: ${itemsJson}`);
}
}
function randomTag(items) {
const tags = [];
for (let i = 0; i < items.length; i++) {
if (items[i].type_ != 5) continue;
tags.push(items[i]);
}
return randomElement(tags);
}
function randomNote(items) {
const notes = [];
for (let i = 0; i < items.length; i++) {
if (items[i].type_ != 1) continue;
notes.push(items[i]);
}
return randomElement(notes);
}
async function execRandomCommand(client) {
const possibleCommands = [
['mkbook {word}', 40], // CREATE FOLDER
['mknote {word}', 70], // CREATE NOTE
[
async () => {
// DELETE RANDOM ITEM
const items = await clientItems(client);
const item = randomElement(items);
if (!item) return;
if (item.type_ == 1) {
return execCommand(client, `rm -f ${item.id}`);
} else if (item.type_ == 2) {
return execCommand(client, `rm -r -f ${item.id}`);
} else if (item.type_ == 5) {
// tag
} else {
throw new Error(`Unknown type: ${item.type_}`);
}
},
30,
],
[
async () => {
// SYNC
const avgSyncDuration = averageSyncDuration();
const options = {};
if (!isNaN(avgSyncDuration)) {
if (Math.random() >= 0.5) {
options.killAfter = avgSyncDuration * Math.random();
}
}
return execCommand(client, 'sync --random-failures', options);
},
30,
],
[
async () => {
// UPDATE RANDOM ITEM
const items = await clientItems(client);
const item = randomNote(items);
if (!item) return;
return execCommand(client, `set ${item.id} title "${randomWord()}"`);
},
50,
],
[
async () => {
// ADD TAG
const items = await clientItems(client);
const note = randomNote(items);
if (!note) return;
const tag = randomTag(items);
const tagTitle = !tag || Math.random() >= 0.9 ? `tag-${randomWord()}` : tag.title;
return execCommand(client, `tag add ${tagTitle} ${note.id}`);
},
50,
],
];
let cmd = null;
while (true) {
cmd = randomElement(possibleCommands);
const r = 1 + Math.floor(Math.random() * 100);
if (r <= cmd[1]) break;
}
cmd = cmd[0];
if (typeof cmd === 'function') {
return cmd();
} else {
cmd = cmd.replace('{word}', randomWord());
return execCommand(client, cmd);
}
}
function averageSyncDuration() {
return lodash.mean(syncDurations);
}
function randomNextCheckTime() {
const output = time.unixMs() + 1000 + Math.random() * 1000 * 120;
logger.info(`Next sync check: ${time.unixMsToIso(output)} (${Math.round((output - time.unixMs()) / 1000)} sec.)`);
return output;
}
function findItem(items, itemId) {
for (let i = 0; i < items.length; i++) {
if (items[i].id == itemId) return items[i];
}
return null;
}
function compareItems(item1, item2) {
const output = [];
for (const n in item1) {
if (!item1.hasOwnProperty(n)) continue;
const p1 = item1[n];
const p2 = item2[n];
if (n == 'notes_') {
p1.sort();
p2.sort();
if (JSON.stringify(p1) !== JSON.stringify(p2)) {
output.push(n);
}
} else {
if (p1 !== p2) output.push(n);
}
}
return output;
}
function findMissingItems_(items1, items2) {
const output = [];
for (let i = 0; i < items1.length; i++) {
const item1 = items1[i];
let found = false;
for (let j = 0; j < items2.length; j++) {
const item2 = items2[j];
if (item1.id == item2.id) {
found = true;
break;
}
}
if (!found) {
output.push(item1);
}
}
return output;
}
function findMissingItems(items1, items2) {
return [findMissingItems_(items1, items2), findMissingItems_(items2, items1)];
}
async function compareClientItems(clientItems) {
const itemCounts = [];
for (let i = 0; i < clientItems.length; i++) {
const items = clientItems[i];
itemCounts.push(items.length);
}
logger.info(`Item count: ${itemCounts.join(', ')}`);
const missingItems = findMissingItems(clientItems[0], clientItems[1]);
if (missingItems[0].length || missingItems[1].length) {
logger.error('Items are different');
logger.error(missingItems);
process.exit(1);
}
const differences = [];
const items = clientItems[0];
for (let i = 0; i < items.length; i++) {
const item1 = items[i];
for (let clientId = 1; clientId < clientItems.length; clientId++) {
const item2 = findItem(clientItems[clientId], item1.id);
if (!item2) {
logger.error(`Item not found on client ${clientId}:`);
logger.error(item1);
process.exit(1);
}
const diff = compareItems(item1, item2);
if (diff.length) {
differences.push({
item1: JSON.stringify(item1),
item2: JSON.stringify(item2),
});
}
}
}
if (differences.length) {
logger.error('Found differences between items:');
logger.error(differences);
process.exit(1);
}
}
async function main() {
await fs.remove(syncDir);
const clients = await createClients();
let clientId = 0;
for (let i = 0; i < clients.length; i++) {
clients[i].activeCommandCount = 0;
}
function handleCommand(clientId) {
if (clients[clientId].activeCommandCount >= 1) return;
clients[clientId].activeCommandCount++;
execRandomCommand(clients[clientId])
.catch(error => {
logger.info(`Client ${clientId}:`);
logger.error(error);
})
.then(r => {
if (r) {
logger.info(`Client ${clientId}:\n${r.trim()}`);
}
clients[clientId].activeCommandCount--;
});
}
let nextSyncCheckTime = randomNextCheckTime();
let state = 'commands';
setInterval(async () => {
if (state == 'waitForSyncCheck') return;
if (state == 'syncCheck') {
state = 'waitForSyncCheck';
const clientItems = [];
// Up to 3 sync operations must be performed by each clients in order for them
// to be perfectly in sync - in order for each items to send their changes
// and get those from the other clients, and to also get changes that are
// made as a result of a sync operation (eg. renaming a folder that conflicts
// with another one).
for (let loopCount = 0; loopCount < 3; loopCount++) {
for (let i = 0; i < clients.length; i++) {
const beforeTime = time.unixMs();
await execCommand(clients[i], 'sync');
syncDurations.push(time.unixMs() - beforeTime);
if (syncDurations.length > 20) syncDurations.splice(0, 1);
if (loopCount === 2) {
const dump = await execCommand(clients[i], 'dump');
clientItems[i] = JSON.parse(dump);
}
}
}
await compareClientItems(clientItems);
nextSyncCheckTime = randomNextCheckTime();
// eslint-disable-next-line require-atomic-updates
state = 'commands';
return;
}
if (state == 'waitForClients') {
for (let i = 0; i < clients.length; i++) {
if (clients[i].activeCommandCount > 0) return;
}
state = 'syncCheck';
return;
}
if (state == 'commands') {
if (nextSyncCheckTime <= time.unixMs()) {
state = 'waitForClients';
return;
}
handleCommand(clientId);
clientId++;
if (clientId >= clients.length) clientId = 0;
}
}, 100);
}
main(process.argv).catch(error => {
logger.error(error);
});