Helps enable app specific compile and run time type checking when using Qlik fields and variables in your Engine-API or Mashup typescript projects
# npm
npm install qlik-ts-ident-checker
# or yarn
yarn add qlik-ts-ident-checker
A very small library (< 1KB gzipped + minified) with no dependencies that helps enable app specific compile and run time type checking for your typescript Qlik Engine-API/Core or Qlik Sense mashup projects. Can be used in plain javascript projects if you get a bit creative (replace the string enums with plain objects).
This library takes a reference to an Engine-API or Capabilities API app instance and:
- Suggests a field and a variable
string enum
definition to use in your project which repersents all field and variable identifiers defined an app's load script - Can validate those enums at runtime to alert you to changes
- e.g. that a field or variable has been created or dropped by a change in the load script whether by design or by accident
- If these enums are used in inline Qlik expressions within your project, then
updating the enums will create compile time errors in
tsc
, thus highlighting necessary changes that need to be made to reflect the the new load script
- The only variables it will check are those defined in the load script or
reserved by the system as script or system variables
- Variables created during a session will not be counted
- Numeric field or variable identifiers will apppear with underscore padding
as identifers in typescript enums cannot be numeric.
- The value they map to will remain unchanged.
- e.g.
'_2000_' = '2000'
/* loadScriptIdentifiers.ts */
import { checkAppIdentifiers } from 'qlik-ts-ident-checker';
// Field enum suggested from a previous run, can be empty initially
export enum Field {
'UNIT VALUE' = 'UNIT VALUE',
// ...
}
// Variable enum suggested from a previous run, can be empty initially
export enum Variable {
'vCurrentMonth' = 'vCurrentMonth',
// ...
}
// Export function to validate the enums at runtime
export async function validate(app: any) {
const result = await checkAppIdentifiers('enigma', app, Field, Variable);
if (!result.fields.matching) {
console.warn('Mismatch between qlik and project fields', result.fields);
}
if (!result.variables.matching) {
console.warn(
'Mismatch between qlik and project variables',
result.variables,
);
}
}
/* somewhere.ts */
import { validate } from './loadScriptIdentifiers.ts';
// ...
const app = await qix.openDoc('super-important-client-project');
// I suggest you only validate in development builds
if (development) {
validate(app);
}
/* somewhereElse.ts */
import { Variable } from './loadScriptIdentifiers.ts';
// ...
const currentMonth = await app.evaluate(`$(${Variable.vCurrentMonth})`);
/**
* Result of validation checks
*/
interface IdentifierCheckResult {
fields: IdentifierKindInfo;
variables: IdentifierKindInfo;
}
interface IdentifierKindInfo {
/** If true, then optionally provided enum matches current data model */
matching: boolean;
/** Suggested new enum definition that matches current data model */
newEnumString: string;
/** List of identifers in current data model */
identifiersInQlik: string[];
/** List of identifers in optionally provided enum */
identifiersInEnum: string[];
/** List of identifiers in current data model, but not within optionally provided enum */
inQlikNotInEnum: string[];
/** List of identifiers in optionally provided enum, but not within the current data model */
inEnumNotInQlik: string[];
}
/**
* Type repersenting a typescript string enum
*/
declare type StringEnum<K extends string> =
| Record<K, string>
| {
[key: string]: string;
};
/**
*
* @param appType indicate whether using the Engine API via enigma or the Capabilities API
* @param app an app instance that corresponds to the appType
* @param fields optionally provide an enum repersenting the fields available in the data model to validate
* @param variables optionally provide an enum repersenting the variables available in the data model to validate
*/
export declare function checkAppIdentifiers<
K1 extends string,
K2 extends string
>(
appType: 'enigma' | 'capabilities',
app: any,
fields?: StringEnum<K1>,
variables?: StringEnum<K2>,
): Promise<IdentifierCheckResult>;