Compare commits
6 Commits
Author | SHA1 | Date | |
---|---|---|---|
e6636d373b | |||
811d797e1b | |||
0909c4f648 | |||
8384d6a299 | |||
19251244d2 | |||
51549db3d9 |
4
package-lock.json
generated
4
package-lock.json
generated
@ -1,12 +1,12 @@
|
|||||||
{
|
{
|
||||||
"name": "@ztimson/utils",
|
"name": "@ztimson/utils",
|
||||||
"version": "0.15.6",
|
"version": "0.16.3",
|
||||||
"lockfileVersion": 3,
|
"lockfileVersion": 3,
|
||||||
"requires": true,
|
"requires": true,
|
||||||
"packages": {
|
"packages": {
|
||||||
"": {
|
"": {
|
||||||
"name": "@ztimson/utils",
|
"name": "@ztimson/utils",
|
||||||
"version": "0.15.6",
|
"version": "0.16.3",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@types/jest": "^29.5.12",
|
"@types/jest": "^29.5.12",
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@ztimson/utils",
|
"name": "@ztimson/utils",
|
||||||
"version": "0.16.0",
|
"version": "0.16.6",
|
||||||
"description": "Utility library",
|
"description": "Utility library",
|
||||||
"author": "Zak Timson",
|
"author": "Zak Timson",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
|
127
src/cache.ts
Normal file
127
src/cache.ts
Normal file
@ -0,0 +1,127 @@
|
|||||||
|
/**
|
||||||
|
* Map of data which tracks whether it is a complete collection & offers optional expiry of cached values
|
||||||
|
*/
|
||||||
|
export class Cache<K, T> {
|
||||||
|
private store: any = {};
|
||||||
|
|
||||||
|
/** Whether cache is complete */
|
||||||
|
complete = false;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create new cache
|
||||||
|
*
|
||||||
|
* @param {keyof T} key Default property to use as primary key
|
||||||
|
* @param {number} ttl Default expiry in milliseconds
|
||||||
|
*/
|
||||||
|
constructor(public readonly key: keyof T, public ttl?: number) {
|
||||||
|
return new Proxy(this, {
|
||||||
|
get: (target: this, prop: string | symbol) => {
|
||||||
|
if(prop in target) return (<any>target)[prop];
|
||||||
|
return target.store[prop];
|
||||||
|
},
|
||||||
|
set: (target: any, prop: string | symbol, value: T) => {
|
||||||
|
if(prop in target) target[prop] = value;
|
||||||
|
else target.store[prop] = value;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
private getKey(value: T): K {
|
||||||
|
return <K>value[this.key];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get all cached items
|
||||||
|
*
|
||||||
|
* @return {T[]} Array of items
|
||||||
|
*/
|
||||||
|
all(): T[] {
|
||||||
|
return Object.values(this.store);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Add a new item to the cache. Like set, but finds key automatically
|
||||||
|
*
|
||||||
|
* @param {T} value Item to add to cache
|
||||||
|
* @param {number | undefined} ttl Override default expiry
|
||||||
|
* @return {this}
|
||||||
|
*/
|
||||||
|
add(value: T, ttl = this.ttl): this {
|
||||||
|
const key = this.getKey(value);
|
||||||
|
this.set(key, value, ttl);
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Add several rows to the cache
|
||||||
|
*
|
||||||
|
* @param {T[]} rows Several items that will be cached using the default key
|
||||||
|
* @param complete Mark cache as complete & reliable, defaults to true
|
||||||
|
* @return {this}
|
||||||
|
*/
|
||||||
|
addAll(rows: T[], complete = true): this {
|
||||||
|
rows.forEach(r => this.add(r));
|
||||||
|
this.complete = complete;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Delete an item from the cache
|
||||||
|
*
|
||||||
|
* @param {K} key Item's primary key
|
||||||
|
*/
|
||||||
|
delete(key: K) {
|
||||||
|
delete this.store[key];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Return cache as an array of key-value pairs
|
||||||
|
* @return {[K, T][]} Key-value pairs array
|
||||||
|
*/
|
||||||
|
entries(): [K, T][] {
|
||||||
|
return <[K, T][]>Object.entries(this.store);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get item from the cache
|
||||||
|
* @param {K} key Key to lookup
|
||||||
|
* @return {T} Cached item
|
||||||
|
*/
|
||||||
|
get(key: K): T {
|
||||||
|
return this.store[key];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get a list of cached keys
|
||||||
|
*
|
||||||
|
* @return {K[]} Array of keys
|
||||||
|
*/
|
||||||
|
keys(): K[] {
|
||||||
|
return <K[]>Object.keys(this.store);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Add an item to the cache manually specifying the key
|
||||||
|
*
|
||||||
|
* @param {K} key Key item will be cached under
|
||||||
|
* @param {T} value Item to cache
|
||||||
|
* @param {number | undefined} ttl Override default expiry
|
||||||
|
* @return {this}
|
||||||
|
*/
|
||||||
|
set(key: K, value: T, ttl = this.ttl): this {
|
||||||
|
this.store[key] = value;
|
||||||
|
if(ttl) setTimeout(() => {
|
||||||
|
this.complete = false;
|
||||||
|
this.delete(key);
|
||||||
|
}, ttl);
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get all cached items
|
||||||
|
*
|
||||||
|
* @return {T[]} Array of items
|
||||||
|
*/
|
||||||
|
values = this.all();
|
||||||
|
}
|
26
src/csv.ts
Normal file
26
src/csv.ts
Normal file
@ -0,0 +1,26 @@
|
|||||||
|
import {dotNotation, flattenObj} from './objects.ts';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convert an object to a CSV string
|
||||||
|
*
|
||||||
|
* @param {any[]} target Array of objects to create CSV from
|
||||||
|
* @param {boolean} flatten Should nested object be flattened or treated as values
|
||||||
|
* @return {string} CSV string
|
||||||
|
*/
|
||||||
|
export function csv(target: any[], flatten=true) {
|
||||||
|
const headers = target.reduce((acc, row) => {
|
||||||
|
Object.keys(flatten ? flattenObj(row) : row)
|
||||||
|
.forEach(key => { if(!acc.includes(key)) acc.push(key); });
|
||||||
|
return acc;
|
||||||
|
}, []);
|
||||||
|
return [
|
||||||
|
headers.join(','),
|
||||||
|
...target.map(row => headers.map((h: string) => {
|
||||||
|
const value = dotNotation<any>(row, h);
|
||||||
|
const type = typeof value;
|
||||||
|
if(type == 'string' && value.includes(',')) return `"${value}"`;
|
||||||
|
if(type == 'object') return `"${JSON.stringify(value)}"`;
|
||||||
|
return value;
|
||||||
|
}).join(','))
|
||||||
|
].join('\n');
|
||||||
|
}
|
36
src/files.ts
36
src/files.ts
@ -1,33 +1,35 @@
|
|||||||
|
import {makeArray} from './array.ts';
|
||||||
import {JSONAttemptParse} from './objects.ts';
|
import {JSONAttemptParse} from './objects.ts';
|
||||||
import {PromiseProgress} from './promise-progress';
|
import {PromiseProgress} from './promise-progress';
|
||||||
|
|
||||||
/**
|
|
||||||
* Download a file from a URL
|
|
||||||
*
|
|
||||||
* @param href URL that will be downloaded
|
|
||||||
* @param {string} name Override download name
|
|
||||||
*/
|
|
||||||
export function download(href: any, name?: string) {
|
|
||||||
const a = document.createElement('a');
|
|
||||||
a.href = href;
|
|
||||||
a.download = name || href.split('/').pop();
|
|
||||||
document.body.appendChild(a);
|
|
||||||
a.click();
|
|
||||||
document.body.removeChild(a);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Download blob as a file
|
* Download blob as a file
|
||||||
*
|
*
|
||||||
* @param {Blob} blob File as a blob
|
* @param {Blob} blob File as a blob
|
||||||
* @param {string} name Name blob will be downloaded as
|
* @param {string} name Name blob will be downloaded as
|
||||||
*/
|
*/
|
||||||
export function downloadBlob(blob: Blob, name: string) {
|
export function downloadFile(blob: Blob | string | string[], name: string) {
|
||||||
|
if(!(blob instanceof Blob)) blob = new Blob(makeArray(blob));
|
||||||
const url = URL.createObjectURL(blob);
|
const url = URL.createObjectURL(blob);
|
||||||
download(url, name);
|
downloadUrl(url, name);
|
||||||
URL.revokeObjectURL(url);
|
URL.revokeObjectURL(url);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Download a file from a URL
|
||||||
|
*
|
||||||
|
* @param href URL that will be downloaded
|
||||||
|
* @param {string} name Override download name
|
||||||
|
*/
|
||||||
|
export function downloadUrl(href: any, name?: string) {
|
||||||
|
const a = document.createElement('a');
|
||||||
|
a.href = href;
|
||||||
|
a.download = name || href.split('/').pop();
|
||||||
|
document.body.appendChild(a);
|
||||||
|
a.click();
|
||||||
|
document.body.removeChild(a);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Open filebrowser & return selected file
|
* Open filebrowser & return selected file
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,7 @@
|
|||||||
export * from './array';
|
export * from './array';
|
||||||
export * from './aset';
|
export * from './aset';
|
||||||
|
export * from './cache';
|
||||||
|
export * from './csv';
|
||||||
export * from './files';
|
export * from './files';
|
||||||
export * from './emitter';
|
export * from './emitter';
|
||||||
export * from './errors';
|
export * from './errors';
|
||||||
|
@ -1,3 +1,5 @@
|
|||||||
|
import {dotNotation, flattenObj} from './objects.ts';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* String of all letters
|
* String of all letters
|
||||||
*/
|
*/
|
||||||
@ -18,16 +20,6 @@ const SYMBOL_LIST = '~`!@#$%^&*()_-+={[}]|\\:;"\'<,>.?/';
|
|||||||
*/
|
*/
|
||||||
const CHAR_LIST = LETTER_LIST + NUMBER_LIST + SYMBOL_LIST;
|
const CHAR_LIST = LETTER_LIST + NUMBER_LIST + SYMBOL_LIST;
|
||||||
|
|
||||||
/**
|
|
||||||
* Generate a random hexadecimal value
|
|
||||||
*
|
|
||||||
* @param {number} length Number of hexadecimal place values
|
|
||||||
* @return {string} Hexadecimal number as a string
|
|
||||||
*/
|
|
||||||
export function randomHex(length: number) {
|
|
||||||
return Array(length).fill(null).map(() => Math.round(Math.random() * 0xF).toString(16)).join('');
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Convert number of bytes into a human-readable size
|
* Convert number of bytes into a human-readable size
|
||||||
*
|
*
|
||||||
@ -95,6 +87,16 @@ export function pad(text: any, length: number, char: string = ' ', start = true)
|
|||||||
return text.toString().padEnd(length, char);
|
return text.toString().padEnd(length, char);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Generate a random hexadecimal value
|
||||||
|
*
|
||||||
|
* @param {number} length Number of hexadecimal place values
|
||||||
|
* @return {string} Hexadecimal number as a string
|
||||||
|
*/
|
||||||
|
export function randomHex(length: number) {
|
||||||
|
return Array(length).fill(null).map(() => Math.round(Math.random() * 0xF).toString(16)).join('');
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Generate a string of random characters.
|
* Generate a string of random characters.
|
||||||
*
|
*
|
||||||
|
@ -39,12 +39,12 @@ export function sleep(ms: number): Promise<void> {
|
|||||||
* await sleepUntil(() => loading); // Won't continue until loading flag is false
|
* await sleepUntil(() => loading); // Won't continue until loading flag is false
|
||||||
* ```
|
* ```
|
||||||
*
|
*
|
||||||
* @param {() => boolean} fn Return true to continue
|
* @param {() => boolean | Promise<boolean>} fn Return true to continue
|
||||||
* @param {number} checkInterval Run function ever x milliseconds
|
* @param {number} checkInterval Run function ever x milliseconds
|
||||||
* @return {Promise<void>} Callback when sleep is over
|
* @return {Promise<void>} Callback when sleep is over
|
||||||
*/
|
*/
|
||||||
export async function sleepUntil(fn : () => boolean, checkInterval=100): Promise<void> {
|
export async function sleepUntil(fn : () => boolean | Promise<boolean>, checkInterval = 100): Promise<void> {
|
||||||
while(fn()) await sleep(checkInterval);
|
while(await fn()) await sleep(checkInterval);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
Reference in New Issue
Block a user