diff --git a/dist/index.js b/dist/index.js
index 2b117b2efff32c0c9fe02f52ec27e38f5ea9d814..51087b584ddf8bf3f5e6862bfcb501948d81bae0 100644
--- a/dist/index.js
+++ b/dist/index.js
@@ -9,28 +9,31 @@ var __createBinding = (this && this.__createBinding) || (Object.create ? (functi
 var __exportStar = (this && this.__exportStar) || function(m, exports) {
     for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
 };
+var __importDefault = (this && this.__importDefault) || function (mod) {
+    return (mod && mod.__esModule) ? mod : { "default": mod };
+};
 Object.defineProperty(exports, "__esModule", { value: true });
 exports.SigningService = exports.RKAUtility = exports.QrCodeTemplate = exports.VerificationError = exports.CryptoService = exports.QrCodeDataService = exports.CommonUtils = exports.StatusesService = exports.VerificationService = exports.CloudflareService = void 0;
 __exportStar(require("./types"), exports);
 __exportStar(require("./services/SigningService/types"), exports);
 __exportStar(require("./utils/common"), exports);
 var CloudflareService_1 = require("./services/CloudflareService");
-Object.defineProperty(exports, "CloudflareService", { enumerable: true, get: function () { return CloudflareService_1.default; } });
+Object.defineProperty(exports, "CloudflareService", { enumerable: true, get: function () { return __importDefault(CloudflareService_1).default; } });
 var VerificationService_1 = require("./services/VerificationService");
-Object.defineProperty(exports, "VerificationService", { enumerable: true, get: function () { return VerificationService_1.default; } });
+Object.defineProperty(exports, "VerificationService", { enumerable: true, get: function () { return __importDefault(VerificationService_1).default; } });
 var StatusesService_1 = require("./services/StatusesService");
-Object.defineProperty(exports, "StatusesService", { enumerable: true, get: function () { return StatusesService_1.default; } });
+Object.defineProperty(exports, "StatusesService", { enumerable: true, get: function () { return __importDefault(StatusesService_1).default; } });
 var common_1 = require("./utils/common");
-Object.defineProperty(exports, "CommonUtils", { enumerable: true, get: function () { return common_1.default; } });
+Object.defineProperty(exports, "CommonUtils", { enumerable: true, get: function () { return __importDefault(common_1).default; } });
 var QrCodeDataService_1 = require("./services/QrCodeDataService");
-Object.defineProperty(exports, "QrCodeDataService", { enumerable: true, get: function () { return QrCodeDataService_1.default; } });
+Object.defineProperty(exports, "QrCodeDataService", { enumerable: true, get: function () { return __importDefault(QrCodeDataService_1).default; } });
 var CryptoService_1 = require("./services/CryptoService");
-Object.defineProperty(exports, "CryptoService", { enumerable: true, get: function () { return CryptoService_1.default; } });
+Object.defineProperty(exports, "CryptoService", { enumerable: true, get: function () { return __importDefault(CryptoService_1).default; } });
 var VerificationError_1 = require("./services/VerificationService/VerificationError");
-Object.defineProperty(exports, "VerificationError", { enumerable: true, get: function () { return VerificationError_1.default; } });
+Object.defineProperty(exports, "VerificationError", { enumerable: true, get: function () { return __importDefault(VerificationError_1).default; } });
 var qrCodeTemplateUtils_1 = require("./utils/qrCodeTemplateUtils");
-Object.defineProperty(exports, "QrCodeTemplate", { enumerable: true, get: function () { return qrCodeTemplateUtils_1.default; } });
+Object.defineProperty(exports, "QrCodeTemplate", { enumerable: true, get: function () { return __importDefault(qrCodeTemplateUtils_1).default; } });
 var rka_1 = require("./utils/rka");
-Object.defineProperty(exports, "RKAUtility", { enumerable: true, get: function () { return rka_1.default; } });
+Object.defineProperty(exports, "RKAUtility", { enumerable: true, get: function () { return __importDefault(rka_1).default; } });
 var SigningService_1 = require("./services/SigningService");
-Object.defineProperty(exports, "SigningService", { enumerable: true, get: function () { return SigningService_1.default; } });
+Object.defineProperty(exports, "SigningService", { enumerable: true, get: function () { return __importDefault(SigningService_1).default; } });
diff --git a/dist/services/AeternityService.js b/dist/services/AeternityService.js
index fca65706bfce2d197e36e03ecdefe314ccc7fd6e..bd3a3106941d837813dbdd043dbbfad4a96c0c39 100644
--- a/dist/services/AeternityService.js
+++ b/dist/services/AeternityService.js
@@ -8,8 +8,11 @@ var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, ge
         step((generator = generator.apply(thisArg, _arguments || [])).next());
     });
 };
+var __importDefault = (this && this.__importDefault) || function (mod) {
+    return (mod && mod.__esModule) ? mod : { "default": mod };
+};
 Object.defineProperty(exports, "__esModule", { value: true });
-const axios_1 = require("axios");
+const axios_1 = __importDefault(require("axios"));
 class AeternityService {
     constructor(nodeUrl, compilerUrl, contractBytecode) {
         this._nodeUrl = "";
diff --git a/dist/services/CloudflareService.js b/dist/services/CloudflareService.js
index 030480634d537fb310ea72ec9d667ea3dc02d404..1f4dd9c67fd94a7e410b25b178044bcffae9b9f2 100644
--- a/dist/services/CloudflareService.js
+++ b/dist/services/CloudflareService.js
@@ -8,9 +8,12 @@ var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, ge
         step((generator = generator.apply(thisArg, _arguments || [])).next());
     });
 };
+var __importDefault = (this && this.__importDefault) || function (mod) {
+    return (mod && mod.__esModule) ? mod : { "default": mod };
+};
 Object.defineProperty(exports, "__esModule", { value: true });
-const axios_1 = require("axios");
-const StatusesService_1 = require("./StatusesService");
+const axios_1 = __importDefault(require("axios"));
+const StatusesService_1 = __importDefault(require("./StatusesService"));
 const common_1 = require("../utils/common");
 const STATUS_FILE_PREFIX = "bs";
 class CloudflareService {
diff --git a/dist/services/CryptoService/CryptoServiceNode.js b/dist/services/CryptoService/CryptoServiceNode.js
index 2e13a114c2a9d3bfea1bc5221c807ddefbb50856..6cb225c8493a47c9ae583c8b2f4d1734d85342a4 100644
--- a/dist/services/CryptoService/CryptoServiceNode.js
+++ b/dist/services/CryptoService/CryptoServiceNode.js
@@ -1,4 +1,23 @@
 "use strict";
+var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
+    if (k2 === undefined) k2 = k;
+    Object.defineProperty(o, k2, { enumerable: true, get: function() { return m[k]; } });
+}) : (function(o, m, k, k2) {
+    if (k2 === undefined) k2 = k;
+    o[k2] = m[k];
+}));
+var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
+    Object.defineProperty(o, "default", { enumerable: true, value: v });
+}) : function(o, v) {
+    o["default"] = v;
+});
+var __importStar = (this && this.__importStar) || function (mod) {
+    if (mod && mod.__esModule) return mod;
+    var result = {};
+    if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
+    __setModuleDefault(result, mod);
+    return result;
+};
 var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
     function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
     return new (P || (P = Promise))(function (resolve, reject) {
@@ -8,10 +27,13 @@ var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, ge
         step((generator = generator.apply(thisArg, _arguments || [])).next());
     });
 };
+var __importDefault = (this && this.__importDefault) || function (mod) {
+    return (mod && mod.__esModule) ? mod : { "default": mod };
+};
 Object.defineProperty(exports, "__esModule", { value: true });
 // All functions are async in order to be compatible with the ICryptoService API
-const js_md5_1 = require("js-md5");
-const crypto = require("crypto");
+const js_md5_1 = __importDefault(require("js-md5"));
+const crypto = __importStar(require("crypto"));
 const common_1 = require("../../utils/common");
 const AES_GCM_ALGO = "aes-256-gcm";
 const encryptAESGCM = (data) => __awaiter(void 0, void 0, void 0, function* () {
diff --git a/dist/services/CryptoService/CryptoServiceWeb.js b/dist/services/CryptoService/CryptoServiceWeb.js
index 762144298f03e36b7012b4ea278dba2111dbd8cf..7e77c6e8d5650d69b8cfa3a27b5c1f9047b98ce4 100644
--- a/dist/services/CryptoService/CryptoServiceWeb.js
+++ b/dist/services/CryptoService/CryptoServiceWeb.js
@@ -8,9 +8,12 @@ var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, ge
         step((generator = generator.apply(thisArg, _arguments || [])).next());
     });
 };
+var __importDefault = (this && this.__importDefault) || function (mod) {
+    return (mod && mod.__esModule) ? mod : { "default": mod };
+};
 Object.defineProperty(exports, "__esModule", { value: true });
 exports.verifyRSASignature = void 0;
-const js_md5_1 = require("js-md5");
+const js_md5_1 = __importDefault(require("js-md5"));
 const common_1 = require("../../utils/common");
 function exportKey(key) {
     return __awaiter(this, void 0, void 0, function* () {
@@ -31,7 +34,7 @@ const convertPemToBinary = (pem) => {
     }
     return common_1.base64ToArrayBuffer(encoded);
 };
-exports.verifyRSASignature = (publicKeyPEM, data, signature) => __awaiter(void 0, void 0, void 0, function* () {
+const verifyRSASignature = (publicKeyPEM, data, signature) => __awaiter(void 0, void 0, void 0, function* () {
     const publicKey = yield crypto.subtle.importKey("spki", convertPemToBinary(publicKeyPEM), {
         name: "RSASSA-PKCS1-v1_5",
         hash: "SHA-256",
@@ -41,6 +44,7 @@ exports.verifyRSASignature = (publicKeyPEM, data, signature) => __awaiter(void 0
         hash: "SHA-256",
     }, publicKey, signature, data);
 });
+exports.verifyRSASignature = verifyRSASignature;
 const encryptAESGCM = (data) => __awaiter(void 0, void 0, void 0, function* () {
     const key = yield crypto.subtle.generateKey({
         name: "AES-GCM",
diff --git a/dist/services/CryptoService/index.js b/dist/services/CryptoService/index.js
index 7166789d24364d957ede7760d163da5b1eb6329c..ce863013f23d371a28f88872926e9ac1b53f39f1 100644
--- a/dist/services/CryptoService/index.js
+++ b/dist/services/CryptoService/index.js
@@ -1,7 +1,10 @@
 "use strict";
+var __importDefault = (this && this.__importDefault) || function (mod) {
+    return (mod && mod.__esModule) ? mod : { "default": mod };
+};
 Object.defineProperty(exports, "__esModule", { value: true });
-const CryptoServiceNode_1 = require("./CryptoServiceNode");
-const CryptoServiceWeb_1 = require("./CryptoServiceWeb");
+const CryptoServiceNode_1 = __importDefault(require("./CryptoServiceNode"));
+const CryptoServiceWeb_1 = __importDefault(require("./CryptoServiceWeb"));
 const service = typeof crypto !== "undefined" && crypto.subtle
     ? CryptoServiceWeb_1.default
     : CryptoServiceNode_1.default;
diff --git a/dist/services/QrCodeDataService.js b/dist/services/QrCodeDataService.js
index 272cddad5f772c8a129b3495a3c508e9361caf47..34a1c5c1dc96939fce94d3e6a31ae35aa60fbd74 100644
--- a/dist/services/QrCodeDataService.js
+++ b/dist/services/QrCodeDataService.js
@@ -8,11 +8,14 @@ var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, ge
         step((generator = generator.apply(thisArg, _arguments || [])).next());
     });
 };
+var __importDefault = (this && this.__importDefault) || function (mod) {
+    return (mod && mod.__esModule) ? mod : { "default": mod };
+};
 Object.defineProperty(exports, "__esModule", { value: true });
-const axios_1 = require("axios");
+const axios_1 = __importDefault(require("axios"));
 const qrcode_data_pb_1 = require("../generated/qrcode_data_pb");
 const common_1 = require("../utils/common");
-const CryptoService_1 = require("./CryptoService");
+const CryptoService_1 = __importDefault(require("./CryptoService"));
 const index_1 = require("../index");
 const EmailDataMessageV1 = qrcode_data_pb_1.vereign.protobuf.qrcode_data.EmailData_V1;
 const KeyDataMessageV1 = qrcode_data_pb_1.vereign.protobuf.qrcode_data.KeyData_V1;
diff --git a/dist/services/SigningService/SigningService.js b/dist/services/SigningService/SigningService.js
index c9e46c2e73c82a68181b706e16810dfc7fff0abd..9c172c917cab9ae575368a6a57b7e80134950774 100644
--- a/dist/services/SigningService/SigningService.js
+++ b/dist/services/SigningService/SigningService.js
@@ -1,4 +1,23 @@
 "use strict";
+var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
+    if (k2 === undefined) k2 = k;
+    Object.defineProperty(o, k2, { enumerable: true, get: function() { return m[k]; } });
+}) : (function(o, m, k, k2) {
+    if (k2 === undefined) k2 = k;
+    o[k2] = m[k];
+}));
+var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
+    Object.defineProperty(o, "default", { enumerable: true, value: v });
+}) : function(o, v) {
+    o["default"] = v;
+});
+var __importStar = (this && this.__importStar) || function (mod) {
+    if (mod && mod.__esModule) return mod;
+    var result = {};
+    if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
+    __setModuleDefault(result, mod);
+    return result;
+};
 var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
     function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
     return new (P || (P = Promise))(function (resolve, reject) {
@@ -8,11 +27,14 @@ var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, ge
         step((generator = generator.apply(thisArg, _arguments || [])).next());
     });
 };
+var __importDefault = (this && this.__importDefault) || function (mod) {
+    return (mod && mod.__esModule) ? mod : { "default": mod };
+};
 Object.defineProperty(exports, "__esModule", { value: true });
 const mime_normalizer_1 = require("@vereign/mime-normalizer");
-const png_1 = require("@vereign/png");
-const mime_parser_1 = require("@vereign/mime-parser");
-const rka_1 = require("../../utils/rka");
+const png_1 = __importStar(require("@vereign/png"));
+const mime_parser_1 = __importDefault(require("@vereign/mime-parser"));
+const rka_1 = __importDefault(require("../../utils/rka"));
 const index_1 = require("../../index");
 const severityLevels_1 = require("./severityLevels");
 class SigningService {
diff --git a/dist/services/SigningService/index.js b/dist/services/SigningService/index.js
index 950ad0266b5748df05b9fe2d87aeff5bade18887..29fd2e01694ebc3e935595563177035c03cb17a4 100644
--- a/dist/services/SigningService/index.js
+++ b/dist/services/SigningService/index.js
@@ -1,4 +1,7 @@
 "use strict";
+var __importDefault = (this && this.__importDefault) || function (mod) {
+    return (mod && mod.__esModule) ? mod : { "default": mod };
+};
 Object.defineProperty(exports, "__esModule", { value: true });
-const SigningService_1 = require("./SigningService");
+const SigningService_1 = __importDefault(require("./SigningService"));
 exports.default = SigningService_1.default;
diff --git a/dist/services/SigningService/severityLevels.js b/dist/services/SigningService/severityLevels.js
index a0047837f28dc4035b5652c760963e9660af4696..c437cc203490337fb6de8c7a97ade80d62db5e25 100644
--- a/dist/services/SigningService/severityLevels.js
+++ b/dist/services/SigningService/severityLevels.js
@@ -45,7 +45,7 @@ exports.SEVERITY_MESSAGE_PER_STATE = [
     "CAUTION! This message has been tampered with. Do not click on any links, do not rely on its information. Contact the sender.",
     "The message is authentic, but at least one attachment appears to have been altered. Caution advised for the attachments, do not open!",
 ];
-exports.findSeverityStateIndex = (htmlValid, plainValid, pseudoPlainValid, attachmentsValid) => {
+const findSeverityStateIndex = (htmlValid, plainValid, pseudoPlainValid, attachmentsValid) => {
     let index = -1;
     const verificationState = [
         +htmlValid,
@@ -70,3 +70,4 @@ exports.findSeverityStateIndex = (htmlValid, plainValid, pseudoPlainValid, attac
     }
     return index;
 };
+exports.findSeverityStateIndex = findSeverityStateIndex;
diff --git a/dist/services/VerificationService/VerificationService.js b/dist/services/VerificationService/VerificationService.js
index 104b8107e9dd875cd7d3f4943e5e7da22985672f..bbe98bc604133f5487ef2e78ba1e4ae1d64ca1ba 100644
--- a/dist/services/VerificationService/VerificationService.js
+++ b/dist/services/VerificationService/VerificationService.js
@@ -8,14 +8,17 @@ var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, ge
         step((generator = generator.apply(thisArg, _arguments || [])).next());
     });
 };
+var __importDefault = (this && this.__importDefault) || function (mod) {
+    return (mod && mod.__esModule) ? mod : { "default": mod };
+};
 Object.defineProperty(exports, "__esModule", { value: true });
 exports.BLOCK_DATA_RETRIEVED = exports.MERKLE_TREE_VERIFIED = exports.TRANSACTION_RETRIEVED = void 0;
 const index_1 = require("../../index");
 const EventEmitter = require("eventemitter2");
 const merkletreejs_1 = require("merkletreejs");
-const AeternityService_1 = require("../AeternityService");
-const CloudflareService_1 = require("../CloudflareService");
-const VerificationError_1 = require("./VerificationError");
+const AeternityService_1 = __importDefault(require("../AeternityService"));
+const CloudflareService_1 = __importDefault(require("../CloudflareService"));
+const VerificationError_1 = __importDefault(require("./VerificationError"));
 exports.TRANSACTION_RETRIEVED = "TRANSACTION_RETRIEVED";
 exports.MERKLE_TREE_VERIFIED = "MERKLE_TREE_VERIFIED";
 exports.BLOCK_DATA_RETRIEVED = "BLOCK_DATA_RETRIEVED";
diff --git a/dist/services/VerificationService/index.js b/dist/services/VerificationService/index.js
index 303dd41f6f5aa8677ef8bf5b9d6d8d15b4f88128..23f5fc63eff5fc7e83bdcc242d05c011d1d22f38 100644
--- a/dist/services/VerificationService/index.js
+++ b/dist/services/VerificationService/index.js
@@ -1,4 +1,7 @@
 "use strict";
+var __importDefault = (this && this.__importDefault) || function (mod) {
+    return (mod && mod.__esModule) ? mod : { "default": mod };
+};
 Object.defineProperty(exports, "__esModule", { value: true });
-const VerificationService_1 = require("./VerificationService");
+const VerificationService_1 = __importDefault(require("./VerificationService"));
 exports.default = VerificationService_1.default;
diff --git a/dist/utils/common.js b/dist/utils/common.js
index b288c65816312209127ff0c473b9ec0627c31418..580fef358ff96a4597afbc811802090e4f94d119 100644
--- a/dist/utils/common.js
+++ b/dist/utils/common.js
@@ -2,19 +2,23 @@
 Object.defineProperty(exports, "__esModule", { value: true });
 exports.arrayBufferToHex = exports.decompressData = exports.compressData = exports.ensureUint8Array = exports.ensureArrayBuffer = exports.ensureBase64 = exports.base64ToArrayBuffer = exports.arrayBufferToBase64 = void 0;
 const zlib_min_1 = require("zlibjs/bin/zlib.min");
-exports.arrayBufferToBase64 = (buffer) => {
+const arrayBufferToBase64 = (buffer) => {
     return Buffer.from(buffer).toString("base64");
 };
-exports.base64ToArrayBuffer = (base64) => {
+exports.arrayBufferToBase64 = arrayBufferToBase64;
+const base64ToArrayBuffer = (base64) => {
     return Buffer.from(base64, "base64");
 };
-exports.ensureBase64 = (data) => {
+exports.base64ToArrayBuffer = base64ToArrayBuffer;
+const ensureBase64 = (data) => {
     return typeof data === "string" ? data : exports.arrayBufferToBase64(data);
 };
-exports.ensureArrayBuffer = (data) => {
+exports.ensureBase64 = ensureBase64;
+const ensureArrayBuffer = (data) => {
     return typeof data === "string" ? exports.base64ToArrayBuffer(data) : data;
 };
-exports.ensureUint8Array = (data) => {
+exports.ensureArrayBuffer = ensureArrayBuffer;
+const ensureUint8Array = (data) => {
     return data instanceof Uint8Array
         ? data
         : typeof data === "string"
@@ -23,19 +27,23 @@ exports.ensureUint8Array = (data) => {
                 ? new Uint8Array(data)
                 : data;
 };
-exports.compressData = (binary) => {
+exports.ensureUint8Array = ensureUint8Array;
+const compressData = (binary) => {
     const deflate = new zlib_min_1.Zlib.Deflate(exports.ensureUint8Array(binary));
     return deflate.compress();
 };
-exports.decompressData = (binary) => {
+exports.compressData = compressData;
+const decompressData = (binary) => {
     const inflate = new zlib_min_1.Zlib.Inflate(exports.ensureUint8Array(binary));
     return inflate.decompress();
 };
-exports.arrayBufferToHex = (buffer) => {
+exports.decompressData = decompressData;
+const arrayBufferToHex = (buffer) => {
     return [...new Uint8Array(buffer)]
         .map((b) => b.toString(16).padStart(2, "0"))
         .join("");
 };
+exports.arrayBufferToHex = arrayBufferToHex;
 exports.default = {
     compressData: exports.compressData,
     decompressData: exports.decompressData,
diff --git a/dist/utils/qrCodeTemplateUtils.d.ts b/dist/utils/qrCodeTemplateUtils.d.ts
index 26ba313037201c52bb286870952ee60625eabbfe..e3e8c4fcc2b5ddb72a24040ff3d69733076846f0 100644
--- a/dist/utils/qrCodeTemplateUtils.d.ts
+++ b/dist/utils/qrCodeTemplateUtils.d.ts
@@ -1,4 +1,21 @@
+interface TemplateOptions {
+    src: string;
+    placeholderSize: number;
+}
+/**
+ * https://www.qrcode.com/en/about/error_correction.html#:~:text=QR%20Code%20has%20error%20correction,of%20data%20QR%20Code%20size.
+ */
+declare type CorrectionLevels = "L" | "M" | "Q" | "H";
+interface QrCodeOptions {
+    width?: number;
+    height?: number;
+    logoSrc?: string;
+    logoWidth?: number;
+    logoHeight?: number;
+    correctionLevel?: CorrectionLevels;
+}
 declare const _default: {
     putQrCodeOnChromakeyTemplate: (qrCodeImageBase64: string, templateImageBase64: string, placeholderWidth: number, placeholderHeight: number, scale?: number) => Promise<string>;
+    generateQrCode: (text: string, qrCodeOptions?: QrCodeOptions, templateOptions?: TemplateOptions) => Promise<string>;
 };
 export default _default;
diff --git a/dist/utils/qrCodeTemplateUtils.js b/dist/utils/qrCodeTemplateUtils.js
index d5c8821ed1ae55f137c1494245aee8861d2fd671..d406abcba39d09ba8dd0571a8e15394289f8ee4c 100644
--- a/dist/utils/qrCodeTemplateUtils.js
+++ b/dist/utils/qrCodeTemplateUtils.js
@@ -8,7 +8,13 @@ var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, ge
         step((generator = generator.apply(thisArg, _arguments || [])).next());
     });
 };
+var __importDefault = (this && this.__importDefault) || function (mod) {
+    return (mod && mod.__esModule) ? mod : { "default": mod };
+};
 Object.defineProperty(exports, "__esModule", { value: true });
+const easyqrcodejs_1 = __importDefault(require("easyqrcodejs"));
+const DEFAULT_TEMPLATE = "";
+const DEFAULT_LOGO = "";
 const findChromakeyBoundaries = (width, height, imageData, chromakeyWidth, chromakeyHeight) => {
     const maskSizeX = chromakeyWidth;
     const maskSizeY = chromakeyHeight;
@@ -87,6 +93,55 @@ const findChromakeyBoundaries = (width, height, imageData, chromakeyWidth, chrom
     }
     return { fromL: -1, fromT: -1, toL: -1, toT: -1 };
 };
+const generateQrCode = (text, qrCodeOptions, templateOptions) => __awaiter(void 0, void 0, void 0, function* () {
+    if (typeof document === "undefined") {
+        throw new Error("QR Code generation is not supported in non-browser environments");
+    }
+    const defaultOptions = {
+        width: 256,
+        height: 256,
+        logoSrc: DEFAULT_LOGO,
+        logoWidth: 75,
+        logoHeight: 66,
+        // Use "L" level as we don't need a redundancy
+        correctionLevel: "L",
+    };
+    qrCodeOptions = qrCodeOptions
+        ? Object.assign(defaultOptions, qrCodeOptions)
+        : defaultOptions;
+    const defaultTemplateOptions = {
+        placeholderSize: 260,
+        src: DEFAULT_TEMPLATE,
+    };
+    templateOptions = templateOptions
+        ? Object.assign(defaultTemplateOptions, templateOptions)
+        : defaultTemplateOptions;
+    const container = document.createElement("div");
+    const options = {
+        text,
+        width: qrCodeOptions.width,
+        height: qrCodeOptions.height,
+        colorDark: "#000000",
+        colorLight: "#ffffff",
+        dotScale: 1,
+        PO: "#d51d32",
+        PI: "#d51d32",
+        correctLevel: easyqrcodejs_1.default.CorrectLevel[qrCodeOptions.correctionLevel],
+    };
+    if (qrCodeOptions.logoSrc) {
+        Object.assign(options, {
+            logo: qrCodeOptions.logoSrc,
+            logoBackgroundTransparent: true,
+            logoWidth: qrCodeOptions.logoWidth,
+            logoHeight: qrCodeOptions.logoHeight,
+        });
+    }
+    new easyqrcodejs_1.default(container, options);
+    const canvas = container.getElementsByTagName("canvas")[0];
+    // Add short async action because of the bug with logo not appearing in the resulting image.
+    yield new Promise((resolve) => setTimeout(resolve, 10));
+    return yield putQrCodeOnChromakeyTemplate(canvas.toDataURL(), templateOptions.src, templateOptions.placeholderSize, templateOptions.placeholderSize);
+});
 const putQrCodeOnChromakeyTemplate = (qrCodeImageBase64, templateImageBase64, placeholderWidth, placeholderHeight, scale = 1) => __awaiter(void 0, void 0, void 0, function* () {
     const qrCodeImage = yield loadImage(qrCodeImageBase64);
     const templateImage = yield loadImage(templateImageBase64);
@@ -128,4 +183,5 @@ const loadImage = (imageSrc) => {
 };
 exports.default = {
     putQrCodeOnChromakeyTemplate,
+    generateQrCode,
 };
diff --git a/dist/utils/rka.js b/dist/utils/rka.js
index f4b20a1f5135bc1d6de3a3af33c0d14803f2f23f..6a51a43f7cbdfe34eec2d18ea200a2c55ab185de 100644
--- a/dist/utils/rka.js
+++ b/dist/utils/rka.js
@@ -50,7 +50,7 @@ const searchFingerprintInText = (text, keywordHashHexString, size) => {
     }
     return -1;
 };
-exports.findRKASubstring = (string, rabinFingerprint, substringSize) => {
+const findRKASubstring = (string, rabinFingerprint, substringSize) => {
     if (!string.length) {
         return "";
     }
@@ -60,6 +60,7 @@ exports.findRKASubstring = (string, rabinFingerprint, substringSize) => {
     }
     return string.slice(startingIndex, startingIndex + substringSize);
 };
+exports.findRKASubstring = findRKASubstring;
 /**
  * @deprecated Will be deleted in version 1.0.1. Use getRabinFingerprint instead.
  */
diff --git a/package.json b/package.json
index 9968eed5c353fd720a113d8db45cfccba0c50d67..6a037af77368c9b70909565d81990c10354ce686 100644
--- a/package.json
+++ b/package.json
@@ -40,6 +40,7 @@
     "@vereign/mime-parser": "git+ssh://git@code.vereign.com:code/js-toolbox/mime-parser.git",
     "@vereign/png": "git+ssh://git@code.vereign.com:code/js-toolbox/png.git",
     "axios": "^0.20.0",
+    "easyqrcodejs": "^4.3.1",
     "eventemitter2": "^6.4.3",
     "google-protobuf": "^3.13.0",
     "js-md5": "^0.7.3",
diff --git a/src/utils/qrCodeTemplateUtils.ts b/src/utils/qrCodeTemplateUtils.ts
index c2058afc743df5858e81a39bb0d065cbf307818a..5d0c12882ac1a155bf2e029e81dd165cf86bc643 100644
--- a/src/utils/qrCodeTemplateUtils.ts
+++ b/src/utils/qrCodeTemplateUtils.ts
@@ -1,3 +1,10 @@
+import QRCode from "easyqrcodejs";
+
+const DEFAULT_TEMPLATE =
+  "";
+const DEFAULT_LOGO =
+  "";
+
 interface ChromakeyBoundaries {
   fromL: number;
   fromT: number;
@@ -108,6 +115,98 @@ const findChromakeyBoundaries = (
   return { fromL: -1, fromT: -1, toL: -1, toT: -1 };
 };
 
+interface TemplateOptions {
+  src: string;
+  placeholderSize: number;
+}
+
+/**
+ * https://www.qrcode.com/en/about/error_correction.html#:~:text=QR%20Code%20has%20error%20correction,of%20data%20QR%20Code%20size.
+ */
+type CorrectionLevels = "L" | "M" | "Q" | "H";
+
+interface QrCodeOptions {
+  width?: number;
+  height?: number;
+  logoSrc?: string;
+  logoWidth?: number;
+  logoHeight?: number;
+  correctionLevel?: CorrectionLevels;
+}
+
+const generateQrCode = async (
+  text: string,
+  qrCodeOptions?: QrCodeOptions,
+  templateOptions?: TemplateOptions
+): Promise<string> => {
+  if (typeof document === "undefined") {
+    throw new Error(
+      "QR Code generation is not supported in non-browser environments"
+    );
+  }
+
+  const defaultOptions: QrCodeOptions = {
+    width: 256,
+    height: 256,
+    logoSrc: DEFAULT_LOGO,
+    logoWidth: 75,
+    logoHeight: 66,
+    // Use "L" level as we don't need a redundancy
+    correctionLevel: "L",
+  };
+
+  qrCodeOptions = qrCodeOptions
+    ? Object.assign(defaultOptions, qrCodeOptions)
+    : defaultOptions;
+
+  const defaultTemplateOptions: TemplateOptions = {
+    placeholderSize: 260,
+    src: DEFAULT_TEMPLATE,
+  };
+
+  templateOptions = templateOptions
+    ? Object.assign(defaultTemplateOptions, templateOptions)
+    : defaultTemplateOptions;
+
+  const container = document.createElement("div");
+
+  const options = {
+    text,
+    width: qrCodeOptions.width,
+    height: qrCodeOptions.height,
+    colorDark: "#000000",
+    colorLight: "#ffffff",
+    dotScale: 1,
+
+    PO: "#d51d32",
+    PI: "#d51d32",
+
+    correctLevel: QRCode.CorrectLevel[qrCodeOptions.correctionLevel], // L, M, Q, H
+  };
+
+  if (qrCodeOptions.logoSrc) {
+    Object.assign(options, {
+      logo: qrCodeOptions.logoSrc,
+      logoBackgroundTransparent: true,
+      logoWidth: qrCodeOptions.logoWidth,
+      logoHeight: qrCodeOptions.logoHeight,
+    });
+  }
+
+  new QRCode(container, options);
+  const canvas = container.getElementsByTagName("canvas")[0];
+
+  // Add short async action because of the bug with logo not appearing in the resulting image.
+  await new Promise((resolve) => setTimeout(resolve, 10));
+
+  return await putQrCodeOnChromakeyTemplate(
+    canvas.toDataURL(),
+    templateOptions.src,
+    templateOptions.placeholderSize,
+    templateOptions.placeholderSize
+  );
+};
+
 const putQrCodeOnChromakeyTemplate = async (
   qrCodeImageBase64: string,
   templateImageBase64: string,
@@ -203,4 +302,5 @@ const loadImage = (imageSrc: string): Promise<HTMLImageElement> => {
 
 export default {
   putQrCodeOnChromakeyTemplate,
+  generateQrCode,
 };
diff --git a/tsconfig.json b/tsconfig.json
index e880418b6314b5810989ea19b7e21fbe4241ce79..c8ea31ebbc2ed30d3d86c994d0ff45136164a80d 100644
--- a/tsconfig.json
+++ b/tsconfig.json
@@ -4,7 +4,8 @@
     "target": "es2015",
     "declaration": true,
     "outDir": "./dist",
-    "allowJs": true
+    "allowJs": true,
+    "esModuleInterop": true
   },
   "include": ["src/**/*"]
 }
diff --git a/yarn.lock b/yarn.lock
index 0d7cbb9f0b7794f27471da3e1e4ef52abb80ddbd..6b579f8ee76c32a2a405722911f536059de9510c 100644
--- a/yarn.lock
+++ b/yarn.lock
@@ -866,10 +866,10 @@
     exec-sh "^0.3.2"
     minimist "^1.2.0"
 
-"@eslint/eslintrc@^0.2.2":
-  version "0.2.2"
-  resolved "https://registry.yarnpkg.com/@eslint/eslintrc/-/eslintrc-0.2.2.tgz#d01fc791e2fc33e88a29d6f3dc7e93d0cd784b76"
-  integrity sha512-EfB5OHNYp1F4px/LI/FEnGylop7nOqkQ1LRzCM0KccA2U8tvV8w01KBv37LbO7nW4H+YhKyo2LcJhRwjjV17QQ==
+"@eslint/eslintrc@^0.3.0":
+  version "0.3.0"
+  resolved "https://registry.yarnpkg.com/@eslint/eslintrc/-/eslintrc-0.3.0.tgz#d736d6963d7003b6514e6324bec9c602ac340318"
+  integrity sha512-1JTKgrOKAHVivSvOYw+sJOunkBjUOvjqWk1DPja7ZFhIS2mX/4EgTT8M7eTK9jrKhL/FvXXEbQwIs3pg1xp3dg==
   dependencies:
     ajv "^6.12.4"
     debug "^4.1.1"
@@ -878,7 +878,7 @@
     ignore "^4.0.6"
     import-fresh "^3.2.1"
     js-yaml "^3.13.1"
-    lodash "^4.17.19"
+    lodash "^4.17.20"
     minimatch "^3.0.4"
     strip-json-comments "^3.1.1"
 
@@ -1123,9 +1123,9 @@
   integrity sha1-p3c2C1s5oaLlEG+OhY8v0tBgxXA=
 
 "@sinonjs/commons@^1.7.0":
-  version "1.8.1"
-  resolved "https://registry.yarnpkg.com/@sinonjs/commons/-/commons-1.8.1.tgz#e7df00f98a203324f6dc7cc606cad9d4a8ab2217"
-  integrity sha512-892K+kWUUi3cl+LlqEWIDrhvLgdL79tECi8JZUyq6IviKy/DNhuzCRlbHUjxK89f4ypPMMaFnFuR9Ie6DoIMsw==
+  version "1.8.2"
+  resolved "https://registry.yarnpkg.com/@sinonjs/commons/-/commons-1.8.2.tgz#858f5c4b48d80778fde4b9d541f27edc0d56488b"
+  integrity sha512-sruwd86RJHdsVf/AtBoijDmUqJp3B6hF/DGC23C+JaegnDHaZyewCjoVGTdg3J0uz3Zs7NnIT05OBOmML72lQw==
   dependencies:
     type-detect "4.0.8"
 
@@ -1201,9 +1201,9 @@
     "@types/istanbul-lib-report" "*"
 
 "@types/jsdom@^16.2.4":
-  version "16.2.5"
-  resolved "https://registry.yarnpkg.com/@types/jsdom/-/jsdom-16.2.5.tgz#74ebad438741d249ecb416c5486dcde4217eb66c"
-  integrity sha512-k/ZaTXtReAjwWu0clU0KLS53dyqZnA8mm+jwKFeFrvufXgICp+VNbskETFxKKAguv0pkaEKTax5MaRmvalM+TA==
+  version "16.2.6"
+  resolved "https://registry.yarnpkg.com/@types/jsdom/-/jsdom-16.2.6.tgz#9ddf0521e49be5365797e690c3ba63148e562c29"
+  integrity sha512-yQA+HxknGtW9AkRTNyiSH3OKW5V+WzO8OPTdne99XwJkYC+KYxfNIcoJjeiSqP3V00PUUpFP6Myoo9wdIu78DQ==
   dependencies:
     "@types/node" "*"
     "@types/parse5" "*"
@@ -1220,14 +1220,14 @@
   integrity sha512-5tXH6Bx/kNGd3MgffdmP4dy2Z+G4eaXw0SE81Tq3BNadtnMR5/ySMzX4SLEzHJzSmPNn4HIdpQsBvXMUykr58w==
 
 "@types/node@*":
-  version "14.14.20"
-  resolved "https://registry.yarnpkg.com/@types/node/-/node-14.14.20.tgz#f7974863edd21d1f8a494a73e8e2b3658615c340"
-  integrity sha512-Y93R97Ouif9JEOWPIUyU+eyIdyRqQR0I8Ez1dzku4hDx34NWh4HbtIc3WNzwB1Y9ULvNGeu5B8h8bVL5cAk4/A==
+  version "14.14.21"
+  resolved "https://registry.yarnpkg.com/@types/node/-/node-14.14.21.tgz#d934aacc22424fe9622ebf6857370c052eae464e"
+  integrity sha512-cHYfKsnwllYhjOzuC5q1VpguABBeecUp24yFluHpn/BQaVxB1CuQ1FSRZCzrPxrkIfWISXV2LbeoBthLWg0+0A==
 
 "@types/node@^13.7.0":
-  version "13.13.39"
-  resolved "https://registry.yarnpkg.com/@types/node/-/node-13.13.39.tgz#237d071fb593d3aaa96f655a8eb2f91e0fb1480c"
-  integrity sha512-wct+WgRTTkBm2R3vbrFOqyZM5w0g+D8KnhstG9463CJBVC3UVZHMToge7iMBR1vDl/I+NWFHUeK9X+JcF0rWKw==
+  version "13.13.40"
+  resolved "https://registry.yarnpkg.com/@types/node/-/node-13.13.40.tgz#f655ef327362cc83912f2e69336ddc62a24a9f88"
+  integrity sha512-eKaRo87lu1yAXrzEJl0zcJxfUMDT5/mZalFyOkT44rnQps41eS2pfWzbaulSPpQLFNy29bFqn+Y5lOTL8ATlEQ==
 
 "@types/normalize-package-data@^2.4.0":
   version "2.4.0"
@@ -1240,9 +1240,9 @@
   integrity sha512-//oorEZjL6sbPcKUaCdIGlIUeH26mgzimjBB77G6XRgnDl/L5wOnpyBGRe/Mmf5CVW3PwEBE1NjiMZ/ssFh4wA==
 
 "@types/parse5@*":
-  version "5.0.3"
-  resolved "https://registry.yarnpkg.com/@types/parse5/-/parse5-5.0.3.tgz#e7b5aebbac150f8b5fdd4a46e7f0bd8e65e19109"
-  integrity sha512-kUNnecmtkunAoQ3CnjmMkzNU/gtxG8guhi+Fk2U/kOpIKjIMKnXGp4IJCgQJrXSgMsWYimYG4TGjz/UzbGEBTw==
+  version "6.0.0"
+  resolved "https://registry.yarnpkg.com/@types/parse5/-/parse5-6.0.0.tgz#38590dc2c3cf5717154064e3ee9b6947ee21b299"
+  integrity sha512-oPwPSj4a1wu9rsXTEGIJz91ISU725t0BmSnUhb57sI+M8XEmvUop84lzuiYdq0Y5M6xLY8DBPg0C2xEQKLyvBA==
 
 "@types/prettier@^2.0.0":
   version "2.1.6"
@@ -1668,7 +1668,7 @@ browser-process-hrtime@^1.0.0:
   resolved "https://registry.yarnpkg.com/browser-process-hrtime/-/browser-process-hrtime-1.0.0.tgz#3c9b4b7d782c8121e56f10106d84c0d0ffc94626"
   integrity sha512-9o5UecI3GhkpM6DrXr69PblIuWxPKk9Y0jHBRhdocZ2y7YECBFCsHm79Pr3OyR2AvjhDkabFJaDJMYRazHgsow==
 
-browserslist@^4.14.5, browserslist@^4.16.0:
+browserslist@^4.14.5, browserslist@^4.16.1:
   version "4.16.1"
   resolved "https://registry.yarnpkg.com/browserslist/-/browserslist-4.16.1.tgz#bf757a2da376b3447b800a16f0f1c96358138766"
   integrity sha512-UXhDrwqsNcpTYJBTZsbGATDxZbiVDsx6UjpmRUmtnP10pr8wAYr5LgFoEFw9ixriQH2mv/NX2SfGzE/o8GndLA==
@@ -1712,12 +1712,12 @@ cache-base@^1.0.1:
     unset-value "^1.0.0"
 
 call-bind@^1.0.0:
-  version "1.0.0"
-  resolved "https://registry.yarnpkg.com/call-bind/-/call-bind-1.0.0.tgz#24127054bb3f9bdcb4b1fb82418186072f77b8ce"
-  integrity sha512-AEXsYIyyDY3MCzbwdhzG3Jx1R0J2wetQyUynn6dYHAO+bg8l1k7jwZtRv4ryryFs7EP+NDlikJlVe59jr0cM2w==
+  version "1.0.2"
+  resolved "https://registry.yarnpkg.com/call-bind/-/call-bind-1.0.2.tgz#b1d4e89e688119c3c9a903ad30abb2f6a919be3c"
+  integrity sha512-7O+FbCihrB5WGbFYesctwmTKae6rOiIzmz1icreWJ+0aA7LJfuqhEso2T9ncpcFtzMQtzXf2QGGueWJGTYsqrA==
   dependencies:
     function-bind "^1.1.1"
-    get-intrinsic "^1.0.0"
+    get-intrinsic "^1.0.2"
 
 callsites@^3.0.0:
   version "3.1.0"
@@ -1735,9 +1735,9 @@ camelcase@^6.0.0:
   integrity sha512-c7wVvbw3f37nuobQNtgsgG9POC9qMbNuMQmTCqZv23b6MIz0fcYpBiOlv9gEN/hdLdnZTDQhg6e9Dq5M1vKvfg==
 
 caniuse-lite@^1.0.30001173:
-  version "1.0.30001173"
-  resolved "https://registry.yarnpkg.com/caniuse-lite/-/caniuse-lite-1.0.30001173.tgz#3c47bbe3cd6d7a9eda7f50ac016d158005569f56"
-  integrity sha512-R3aqmjrICdGCTAnSXtNyvWYMK3YtV5jwudbq0T7nN9k4kmE4CBuwPqyJ+KBzepSTh0huivV2gLbSMEzTTmfeYw==
+  version "1.0.30001178"
+  resolved "https://registry.yarnpkg.com/caniuse-lite/-/caniuse-lite-1.0.30001178.tgz#3ad813b2b2c7d585b0be0a2440e1e233c6eabdbc"
+  integrity sha512-VtdZLC0vsXykKni8Uztx45xynytOi71Ufx9T8kHptSw9AL4dpqailUJJHavttuzUe1KYuBYtChiWv+BAb7mPmQ==
 
 capture-exit@^2.0.0:
   version "2.0.0"
@@ -1914,11 +1914,11 @@ copy-descriptor@^0.1.0:
   integrity sha1-Z29us8OZl8LuGsOpJP1hJHSPV40=
 
 core-js-compat@^3.8.0:
-  version "3.8.2"
-  resolved "https://registry.yarnpkg.com/core-js-compat/-/core-js-compat-3.8.2.tgz#3717f51f6c3d2ebba8cbf27619b57160029d1d4c"
-  integrity sha512-LO8uL9lOIyRRrQmZxHZFl1RV+ZbcsAkFWTktn5SmH40WgLtSNYN4m4W2v9ONT147PxBY/XrRhrWq8TlvObyUjQ==
+  version "3.8.3"
+  resolved "https://registry.yarnpkg.com/core-js-compat/-/core-js-compat-3.8.3.tgz#9123fb6b9cad30f0651332dc77deba48ef9b0b3f"
+  integrity sha512-1sCb0wBXnBIL16pfFG1Gkvei6UzvKyTNYpiC41yrdjEv0UoJoq9E/abTMzyYJ6JpTkAj15dLjbqifIzEBDVvog==
   dependencies:
-    browserslist "^4.16.0"
+    browserslist "^4.16.1"
     semver "7.0.0"
 
 core-util-is@1.0.2:
@@ -2097,6 +2097,11 @@ domexception@^2.0.1:
   dependencies:
     webidl-conversions "^5.0.0"
 
+easyqrcodejs@^4.3.1:
+  version "4.3.1"
+  resolved "https://registry.yarnpkg.com/easyqrcodejs/-/easyqrcodejs-4.3.1.tgz#52eb768824dfbbad09131a515287474a2c3f1aa1"
+  integrity sha512-/1QbzGZr60zjGmOyry5ZxXyNMFxVpjEHyU1nirKqy/YhfiztODb9nP+ev7EO70xjm/KoviE8rVTbHY96QryUMA==
+
 ecc-jsbn@~0.1.1:
   version "0.1.2"
   resolved "https://registry.yarnpkg.com/ecc-jsbn/-/ecc-jsbn-0.1.2.tgz#3a83a904e54353287874c564b7549386849a98c9"
@@ -2106,9 +2111,9 @@ ecc-jsbn@~0.1.1:
     safer-buffer "^2.1.0"
 
 electron-to-chromium@^1.3.634:
-  version "1.3.634"
-  resolved "https://registry.yarnpkg.com/electron-to-chromium/-/electron-to-chromium-1.3.634.tgz#82ea400f520f739c4f6ff00c1f7524827a917d25"
-  integrity sha512-QPrWNYeE/A0xRvl/QP3E0nkaEvYUvH3gM04ZWYtIa6QlSpEetRlRI1xvQ7hiMIySHHEV+mwDSX8Kj4YZY6ZQAw==
+  version "1.3.641"
+  resolved "https://registry.yarnpkg.com/electron-to-chromium/-/electron-to-chromium-1.3.641.tgz#03f14efd70a7971eff2efc947b3c1d0f717c82b9"
+  integrity sha512-b0DLhsHSHESC1I+Nx6n4w4Lr61chMd3m/av1rZQhS2IXTzaS5BMM5N+ldWdMIlni9CITMRM09m8He4+YV/92TA==
 
 emittery@^0.7.1:
   version "0.7.2"
@@ -2199,12 +2204,12 @@ eslint-visitor-keys@^2.0.0:
   integrity sha512-QudtT6av5WXels9WjIM7qz1XD1cWGvX4gGXvp/zBn9nXG02D0utdU3Em2m/QjTnrsk6bBjmCygl3rmj118msQQ==
 
 eslint@^7.7.0:
-  version "7.17.0"
-  resolved "https://registry.yarnpkg.com/eslint/-/eslint-7.17.0.tgz#4ccda5bf12572ad3bf760e6f195886f50569adb0"
-  integrity sha512-zJk08MiBgwuGoxes5sSQhOtibZ75pz0J35XTRlZOk9xMffhpA9BTbQZxoXZzOl5zMbleShbGwtw+1kGferfFwQ==
+  version "7.18.0"
+  resolved "https://registry.yarnpkg.com/eslint/-/eslint-7.18.0.tgz#7fdcd2f3715a41fe6295a16234bd69aed2c75e67"
+  integrity sha512-fbgTiE8BfUJZuBeq2Yi7J3RB3WGUQ9PNuNbmgi6jt9Iv8qrkxfy19Ds3OpL1Pm7zg3BtTVhvcUZbIRQ0wmSjAQ==
   dependencies:
     "@babel/code-frame" "^7.0.0"
-    "@eslint/eslintrc" "^0.2.2"
+    "@eslint/eslintrc" "^0.3.0"
     ajv "^6.10.0"
     chalk "^4.0.0"
     cross-spawn "^7.0.2"
@@ -2228,7 +2233,7 @@ eslint@^7.7.0:
     js-yaml "^3.13.1"
     json-stable-stringify-without-jsonify "^1.0.1"
     levn "^0.4.1"
-    lodash "^4.17.19"
+    lodash "^4.17.20"
     minimatch "^3.0.4"
     natural-compare "^1.4.0"
     optionator "^0.9.1"
@@ -2457,12 +2462,20 @@ find-up@^4.0.0, find-up@^4.1.0:
     locate-path "^5.0.0"
     path-exists "^4.0.0"
 
-find-versions@^3.2.0:
-  version "3.2.0"
-  resolved "https://registry.yarnpkg.com/find-versions/-/find-versions-3.2.0.tgz#10297f98030a786829681690545ef659ed1d254e"
-  integrity sha512-P8WRou2S+oe222TOCHitLy8zj+SIsVJh52VP4lvXkaFVnOFFdoWv1H1Jjvel1aI6NCFOAaeAVm8qrI0odiLcww==
+find-up@^5.0.0:
+  version "5.0.0"
+  resolved "https://registry.yarnpkg.com/find-up/-/find-up-5.0.0.tgz#4c92819ecb7083561e4f4a240a86be5198f536fc"
+  integrity sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==
   dependencies:
-    semver-regex "^2.0.0"
+    locate-path "^6.0.0"
+    path-exists "^4.0.0"
+
+find-versions@^4.0.0:
+  version "4.0.0"
+  resolved "https://registry.yarnpkg.com/find-versions/-/find-versions-4.0.0.tgz#3c57e573bf97769b8cb8df16934b627915da4965"
+  integrity sha512-wgpWy002tA+wgmO27buH/9KzyEOQnKsG/R0yrcjPT9BOFm0zRBVQbZ95nRGXWMywS8YR5knRbpohio0bcJABxQ==
+  dependencies:
+    semver-regex "^3.1.2"
 
 flat-cache@^3.0.4:
   version "3.0.4"
@@ -2543,7 +2556,7 @@ get-caller-file@^2.0.1:
   resolved "https://registry.yarnpkg.com/get-caller-file/-/get-caller-file-2.0.5.tgz#4f94412a82db32f36e3b0b9741f8a97feb031f7e"
   integrity sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==
 
-get-intrinsic@^1.0.0:
+get-intrinsic@^1.0.2:
   version "1.0.2"
   resolved "https://registry.yarnpkg.com/get-intrinsic/-/get-intrinsic-1.0.2.tgz#6820da226e50b24894e08859469dc68361545d49"
   integrity sha512-aeX0vrFm21ILl3+JpFFRNe9aUvp6VFZb2/CTbgLb8j75kOhvoNYjt9d8KA/tJG4gSo8nzEDedRl0h7vDmBYRVg==
@@ -2732,17 +2745,17 @@ human-signals@^1.1.1:
   integrity sha512-SEQu7vl8KjNL2eoGBLF3+wAjpsNfA9XMlXAYj/3EdaNfAlxKthD1xjEQfGOUhllCGGJVNY34bRr6lPINhNjyZw==
 
 husky@^4.2.5:
-  version "4.3.6"
-  resolved "https://registry.yarnpkg.com/husky/-/husky-4.3.6.tgz#ebd9dd8b9324aa851f1587318db4cccb7665a13c"
-  integrity sha512-o6UjVI8xtlWRL5395iWq9LKDyp/9TE7XMOTvIpEVzW638UcGxTmV5cfel6fsk/jbZSTlvfGVJf2svFtybcIZag==
+  version "4.3.8"
+  resolved "https://registry.yarnpkg.com/husky/-/husky-4.3.8.tgz#31144060be963fd6850e5cc8f019a1dfe194296d"
+  integrity sha512-LCqqsB0PzJQ/AlCgfrfzRe3e3+NvmefAdKQhRYpxS4u6clblBoDdzzvHi8fmxKRzvMxPY/1WZWzomPZww0Anow==
   dependencies:
     chalk "^4.0.0"
     ci-info "^2.0.0"
     compare-versions "^3.6.0"
     cosmiconfig "^7.0.0"
-    find-versions "^3.2.0"
+    find-versions "^4.0.0"
     opencollective-postinstall "^2.0.2"
-    pkg-dir "^4.2.0"
+    pkg-dir "^5.0.0"
     please-upgrade-node "^3.2.0"
     slash "^3.0.0"
     which-pm-runs "^1.0.0"
@@ -3660,6 +3673,13 @@ locate-path@^5.0.0:
   dependencies:
     p-locate "^4.1.0"
 
+locate-path@^6.0.0:
+  version "6.0.0"
+  resolved "https://registry.yarnpkg.com/locate-path/-/locate-path-6.0.0.tgz#55321eb309febbc59c4801d931a72452a681d286"
+  integrity sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==
+  dependencies:
+    p-locate "^5.0.0"
+
 lodash.sortby@^4.7.0:
   version "4.7.0"
   resolved "https://registry.yarnpkg.com/lodash.sortby/-/lodash.sortby-4.7.0.tgz#edd14c824e2cc9c1e0b0a1b42bb5210516a42438"
@@ -3870,9 +3890,9 @@ node-notifier@^8.0.0:
     which "^2.0.2"
 
 node-releases@^1.1.69:
-  version "1.1.69"
-  resolved "https://registry.yarnpkg.com/node-releases/-/node-releases-1.1.69.tgz#3149dbde53b781610cd8b486d62d86e26c3725f6"
-  integrity sha512-DGIjo79VDEyAnRlfSqYTsy+yoHd2IOjJiKUozD2MV2D85Vso6Bug56mb9tT/fY5Urt0iqk01H7x+llAruDR2zA==
+  version "1.1.70"
+  resolved "https://registry.yarnpkg.com/node-releases/-/node-releases-1.1.70.tgz#66e0ed0273aa65666d7fe78febe7634875426a08"
+  integrity sha512-Slf2s69+2/uAD79pVVQo8uSiC34+g8GWY8UH2Qtqv34ZfhYrxpYpfzs9Js9d6O0mbDmALuxaTlplnBTnSELcrw==
 
 normalize-package-data@^2.5.0:
   version "2.5.0"
@@ -4018,6 +4038,13 @@ p-limit@^2.2.0:
   dependencies:
     p-try "^2.0.0"
 
+p-limit@^3.0.2:
+  version "3.1.0"
+  resolved "https://registry.yarnpkg.com/p-limit/-/p-limit-3.1.0.tgz#e1daccbe78d0d1388ca18c64fea38e3e57e3706b"
+  integrity sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==
+  dependencies:
+    yocto-queue "^0.1.0"
+
 p-locate@^4.1.0:
   version "4.1.0"
   resolved "https://registry.yarnpkg.com/p-locate/-/p-locate-4.1.0.tgz#a3428bb7088b3a60292f66919278b7c297ad4f07"
@@ -4025,6 +4052,13 @@ p-locate@^4.1.0:
   dependencies:
     p-limit "^2.2.0"
 
+p-locate@^5.0.0:
+  version "5.0.0"
+  resolved "https://registry.yarnpkg.com/p-locate/-/p-locate-5.0.0.tgz#83c8315c6785005e3bd021839411c9e110e6d834"
+  integrity sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==
+  dependencies:
+    p-limit "^3.0.2"
+
 p-map@^4.0.0:
   version "4.0.0"
   resolved "https://registry.yarnpkg.com/p-map/-/p-map-4.0.0.tgz#bb2f95a5eda2ec168ec9274e06a747c3e2904d2b"
@@ -4045,9 +4079,9 @@ parent-module@^1.0.0:
     callsites "^3.0.0"
 
 parse-json@^5.0.0:
-  version "5.1.0"
-  resolved "https://registry.yarnpkg.com/parse-json/-/parse-json-5.1.0.tgz#f96088cdf24a8faa9aea9a009f2d9d942c999646"
-  integrity sha512-+mi/lmVVNKFNVyLXV31ERiy2CY5E1/F6QtJFEzoChPRwwngMNXRDQ9GJ5WdE2Z2P4AujsOi0/+2qHID68KwfIQ==
+  version "5.2.0"
+  resolved "https://registry.yarnpkg.com/parse-json/-/parse-json-5.2.0.tgz#c76fc66dee54231c962b22bcc8a72cf2f99753cd"
+  integrity sha512-ayCKvm/phCGxOkYRSCM82iDwct8/EonSEgCSxWxD7ve6jHggsFl4fZVQBPRNgQoKiuV/odhFrGzQXZwbifC8Rg==
   dependencies:
     "@babel/code-frame" "^7.0.0"
     error-ex "^1.3.1"
@@ -4126,6 +4160,13 @@ pkg-dir@^4.2.0:
   dependencies:
     find-up "^4.0.0"
 
+pkg-dir@^5.0.0:
+  version "5.0.0"
+  resolved "https://registry.yarnpkg.com/pkg-dir/-/pkg-dir-5.0.0.tgz#a02d6aebe6ba133a928f74aec20bafdfe6b8e760"
+  integrity sha512-NPE8TDbzl/3YQYY7CSS228s3g2ollTFnc+Qi3tqmqJp9Vg2ovUpixcJEo2HJScN2Ez+kEaal6y70c0ehqJBJeA==
+  dependencies:
+    find-up "^5.0.0"
+
 please-upgrade-node@^3.2.0:
   version "3.2.0"
   resolved "https://registry.yarnpkg.com/please-upgrade-node/-/please-upgrade-node-3.2.0.tgz#aeddd3f994c933e4ad98b99d9a556efa0e2fe942"
@@ -4307,9 +4348,9 @@ regjsgen@^0.5.1:
   integrity sha512-OFFT3MfrH90xIW8OOSyUrk6QHD5E9JOTeGodiJeBS3J6IwlgzJMNE/1bZklWz5oTg+9dCMyEetclvCVXOPoN3A==
 
 regjsparser@^0.6.4:
-  version "0.6.4"
-  resolved "https://registry.yarnpkg.com/regjsparser/-/regjsparser-0.6.4.tgz#a769f8684308401a66e9b529d2436ff4d0666272"
-  integrity sha512-64O87/dPDgfk8/RQqC4gkZoGyyWFIEUTTh80CU6CWuK5vkCGyekIx+oKcEIYtP/RAxSQltCZHCNu/mdd7fqlJw==
+  version "0.6.6"
+  resolved "https://registry.yarnpkg.com/regjsparser/-/regjsparser-0.6.6.tgz#6d8c939d1a654f78859b08ddcc4aa777f3fa800a"
+  integrity sha512-jjyuCp+IEMIm3N1H1LLTJW1EISEJV9+5oHdEyrt43Pg9cDSb6rrLZei2cVWpl0xTjmmlpec/lEQGYgM7xfpGCQ==
   dependencies:
     jsesc "~0.5.0"
 
@@ -4501,10 +4542,10 @@ semver-compare@^1.0.0:
   resolved "https://registry.yarnpkg.com/semver-compare/-/semver-compare-1.0.0.tgz#0dee216a1c941ab37e9efb1788f6afc5ff5537fc"
   integrity sha1-De4hahyUGrN+nvsXiPavxf9VN/w=
 
-semver-regex@^2.0.0:
-  version "2.0.0"
-  resolved "https://registry.yarnpkg.com/semver-regex/-/semver-regex-2.0.0.tgz#a93c2c5844539a770233379107b38c7b4ac9d338"
-  integrity sha512-mUdIBBvdn0PLOeP3TEkMH7HHeUP3GjsXCwKarjv/kGmUFOYg1VqEemKhoQpWMu6X2I8kHeuVdGibLGkVK+/5Qw==
+semver-regex@^3.1.2:
+  version "3.1.2"
+  resolved "https://registry.yarnpkg.com/semver-regex/-/semver-regex-3.1.2.tgz#34b4c0d361eef262e07199dbef316d0f2ab11807"
+  integrity sha512-bXWyL6EAKOJa81XG1OZ/Yyuq+oT0b2YLlxx7c+mrdYPaPbnj6WgVULXhinMIeZGufuUBu/eVRqXEhiv4imfwxA==
 
 "semver@2 || 3 || 4 || 5", semver@^5.4.1, semver@^5.5.0:
   version "5.7.1"
@@ -4951,9 +4992,9 @@ tslib@^1.8.1, tslib@^1.9.0:
   integrity sha512-Xni35NKzjgMrwevysHTCArtLDpPvye8zV/0E4EyYn43P7/7qvQwPh9BGkHewbMulVntbigmcT7rdX3BNo9wRJg==
 
 tsutils@^3.17.1:
-  version "3.18.0"
-  resolved "https://registry.yarnpkg.com/tsutils/-/tsutils-3.18.0.tgz#38add50a28ec97e988cb43c5b32e55d1ff4a222a"
-  integrity sha512-D9Tu8nE3E7D1Bsf/V29oMHceMf+gnVO+pDguk/A5YRo1cLpkiQ48ZnbbS57pvvHeY+OIeNQx1vf4ASPlEtRpcA==
+  version "3.19.1"
+  resolved "https://registry.yarnpkg.com/tsutils/-/tsutils-3.19.1.tgz#d8566e0c51c82f32f9c25a4d367cd62409a547a9"
+  integrity sha512-GEdoBf5XI324lu7ycad7s6laADfnAqCw6wLGI+knxvw9vsIYBaJfYdmeCEG3FMMUiSm3OGgNb+m6utsWf5h9Vw==
   dependencies:
     tslib "^1.8.1"
 
@@ -5057,9 +5098,9 @@ unset-value@^1.0.0:
     isobject "^3.0.0"
 
 uri-js@^4.2.2:
-  version "4.4.0"
-  resolved "https://registry.yarnpkg.com/uri-js/-/uri-js-4.4.0.tgz#aa714261de793e8a82347a7bcc9ce74e86f28602"
-  integrity sha512-B0yRTzYdUCCn9n+F4+Gh4yIDtMQcaJsmYBDsTSG8g/OejKBodLQ2IHfN3bM7jUsRXndopT7OIXWdYqc1fjmV6g==
+  version "4.4.1"
+  resolved "https://registry.yarnpkg.com/uri-js/-/uri-js-4.4.1.tgz#9b1a52595225859e55f669d928f88c6c57f2a77e"
+  integrity sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg==
   dependencies:
     punycode "^2.1.0"
 
@@ -5282,6 +5323,11 @@ yargs@^15.4.1:
     y18n "^4.0.0"
     yargs-parser "^18.1.2"
 
+yocto-queue@^0.1.0:
+  version "0.1.0"
+  resolved "https://registry.yarnpkg.com/yocto-queue/-/yocto-queue-0.1.0.tgz#0294eb3dee05028d31ee1a5fa2c556a6aaf10a1b"
+  integrity sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==
+
 zlibjs@^0.3.1:
   version "0.3.1"
   resolved "https://registry.yarnpkg.com/zlibjs/-/zlibjs-0.3.1.tgz#50197edb28a1c42ca659cc8b4e6a9ddd6d444554"