@macedon-technologies/batman 1.0.16
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.jsii +637 -0
- package/API.md +5671 -0
- package/LICENSE +202 -0
- package/README.md +86 -0
- package/lib/ApiConstructFile.d.ts +5 -0
- package/lib/ApiConstructFile.js +175 -0
- package/lib/ApiFunctionFile.d.ts +5 -0
- package/lib/ApiFunctionFile.js +30 -0
- package/lib/Application.d.ts +21 -0
- package/lib/Application.js +72 -0
- package/lib/ApplicationStackFile.d.ts +5 -0
- package/lib/ApplicationStackFile.js +26 -0
- package/lib/BaseApplicationStackFile.d.ts +5 -0
- package/lib/BaseApplicationStackFile.js +172 -0
- package/lib/BatmanProject.d.ts +25 -0
- package/lib/BatmanProject.js +101 -0
- package/lib/GitHubRolesFile.d.ts +9 -0
- package/lib/GitHubRolesFile.js +26 -0
- package/lib/GitHubRolesStackFile.d.ts +8 -0
- package/lib/GitHubRolesStackFile.js +165 -0
- package/lib/LocalDevAppFile.d.ts +10 -0
- package/lib/LocalDevAppFile.js +34 -0
- package/lib/MainFile.d.ts +24 -0
- package/lib/MainFile.js +64 -0
- package/lib/MainTestFile.d.ts +5 -0
- package/lib/MainTestFile.js +33 -0
- package/lib/PrCleanupWorkflow.d.ts +9 -0
- package/lib/PrCleanupWorkflow.js +100 -0
- package/lib/PrDeployWorkflow.d.ts +10 -0
- package/lib/PrDeployWorkflow.js +106 -0
- package/lib/ProductionDeployWorkflow.d.ts +11 -0
- package/lib/ProductionDeployWorkflow.js +68 -0
- package/lib/StagingDeployWorkflow.d.ts +11 -0
- package/lib/StagingDeployWorkflow.js +68 -0
- package/lib/StaticWebsiteConstructFile.d.ts +5 -0
- package/lib/StaticWebsiteConstructFile.js +198 -0
- package/lib/ViteReactProject.d.ts +45 -0
- package/lib/ViteReactProject.js +654 -0
- package/lib/index.d.ts +5 -0
- package/lib/index.js +20 -0
- package/package.json +125 -0
|
@@ -0,0 +1,654 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.ViteReactProject = void 0;
|
|
4
|
+
const path_1 = require("path");
|
|
5
|
+
const projen_1 = require("projen");
|
|
6
|
+
const typescript_1 = require("projen/lib/typescript");
|
|
7
|
+
/**
|
|
8
|
+
* Vite + React TypeScript project
|
|
9
|
+
*/
|
|
10
|
+
class ViteReactProject extends typescript_1.TypeScriptAppProject {
|
|
11
|
+
constructor(options) {
|
|
12
|
+
const { sampleCode = true, srcdir = 'src', publicDir = 'public', title, ...restOptions } = options;
|
|
13
|
+
super({
|
|
14
|
+
...restOptions,
|
|
15
|
+
sampleCode: false, // We'll handle sample code ourselves
|
|
16
|
+
eslint: false, // We'll configure eslint ourselves
|
|
17
|
+
jest: false, // Vite uses vitest typically
|
|
18
|
+
package: true,
|
|
19
|
+
disableTsconfig: true, // We'll manage tsconfig ourselves
|
|
20
|
+
deps: [
|
|
21
|
+
'@clerk/clerk-react',
|
|
22
|
+
'@tanstack/react-query',
|
|
23
|
+
'@tailwindcss/postcss',
|
|
24
|
+
'autoprefixer',
|
|
25
|
+
'postcss',
|
|
26
|
+
'tailwindcss',
|
|
27
|
+
],
|
|
28
|
+
});
|
|
29
|
+
// Initialize config fields storage
|
|
30
|
+
this.configFields = new Map();
|
|
31
|
+
this.srcdir = srcdir;
|
|
32
|
+
this.publicDir = publicDir;
|
|
33
|
+
// Add Vite component
|
|
34
|
+
new ViteComponent(this, {
|
|
35
|
+
srcdir: this.srcdir,
|
|
36
|
+
});
|
|
37
|
+
// Add ESLint configuration
|
|
38
|
+
new ViteEslintConfig(this);
|
|
39
|
+
// Add index.html
|
|
40
|
+
new IndexHtml(this, {
|
|
41
|
+
title: title ?? this.name,
|
|
42
|
+
srcdir: this.srcdir,
|
|
43
|
+
});
|
|
44
|
+
// Add TypeScript configs
|
|
45
|
+
new ViteTsConfigs(this, {
|
|
46
|
+
srcdir: this.srcdir,
|
|
47
|
+
});
|
|
48
|
+
// Add .gitignore entries
|
|
49
|
+
this.gitignore?.addPatterns('dist', 'dist-ssr', '*.local');
|
|
50
|
+
// Add vite-env.d.ts for type declarations
|
|
51
|
+
new ViteEnvDts(this, {
|
|
52
|
+
srcdir: this.srcdir,
|
|
53
|
+
});
|
|
54
|
+
// Add PostCSS config for Tailwind
|
|
55
|
+
new PostCSSConfigFile(this);
|
|
56
|
+
// Add config.ts for runtime configuration
|
|
57
|
+
new ConfigFile(this, {
|
|
58
|
+
srcdir: this.srcdir,
|
|
59
|
+
});
|
|
60
|
+
// Add sample config.json
|
|
61
|
+
new ConfigJsonFile(this, {});
|
|
62
|
+
// Add sample code if requested
|
|
63
|
+
if (sampleCode) {
|
|
64
|
+
new ViteSampleCode(this, {
|
|
65
|
+
srcdir: this.srcdir,
|
|
66
|
+
publicDir: this.publicDir,
|
|
67
|
+
});
|
|
68
|
+
}
|
|
69
|
+
// Use relative path for devCommand
|
|
70
|
+
const relativeOutdir = this.parent
|
|
71
|
+
? (0, path_1.relative)(this.parent.outdir, this.outdir)
|
|
72
|
+
: (0, path_1.basename)(this.outdir);
|
|
73
|
+
this.devCommand = `cd ${relativeOutdir} && yarn dev`;
|
|
74
|
+
}
|
|
75
|
+
/**
|
|
76
|
+
* Add a custom field to the Config type
|
|
77
|
+
* @param fieldName The name of the field
|
|
78
|
+
* @param fieldType The TypeScript type of the field (e.g., 'string', 'number', 'boolean', 'string[]')
|
|
79
|
+
* @returns this project instance for chaining
|
|
80
|
+
*/
|
|
81
|
+
addConfigField(fieldName, fieldType) {
|
|
82
|
+
this.configFields.set(fieldName, fieldType);
|
|
83
|
+
return this;
|
|
84
|
+
}
|
|
85
|
+
/**
|
|
86
|
+
* Get the config fields map (for internal use by ConfigFile)
|
|
87
|
+
* @internal
|
|
88
|
+
*/
|
|
89
|
+
getConfigFields() {
|
|
90
|
+
return this.configFields;
|
|
91
|
+
}
|
|
92
|
+
}
|
|
93
|
+
exports.ViteReactProject = ViteReactProject;
|
|
94
|
+
/**
|
|
95
|
+
* Configures Vite dependencies and scripts
|
|
96
|
+
*/
|
|
97
|
+
class ViteComponent extends projen_1.Component {
|
|
98
|
+
constructor(project, _options) {
|
|
99
|
+
super(project);
|
|
100
|
+
const tsProject = project;
|
|
101
|
+
// Set package type to module for ESM support
|
|
102
|
+
tsProject.package.addField('type', 'module');
|
|
103
|
+
// Add React dependencies
|
|
104
|
+
tsProject.addDeps('react@^19.2.0', 'react-dom@^19.2.0');
|
|
105
|
+
// Add dev dependencies
|
|
106
|
+
tsProject.addDevDeps('@types/react@^19.2.2', '@types/react-dom@^19.2.2', '@vitejs/plugin-react@^5.1.0', 'vite@^5.4.0', 'typescript@~5.9.3', '@eslint/js@^9.39.1', 'eslint@^9.39.1', 'eslint-plugin-react-hooks@^5.2.0', 'eslint-plugin-react-refresh@^0.4.24', 'globals@^16.5.0', 'typescript-eslint@^8.46.3');
|
|
107
|
+
// Create vite.config.ts as a managed file
|
|
108
|
+
new ViteConfigFile(project);
|
|
109
|
+
// Add scripts
|
|
110
|
+
tsProject.addTask('dev', {
|
|
111
|
+
description: 'Start development server',
|
|
112
|
+
exec: 'vite',
|
|
113
|
+
});
|
|
114
|
+
// Remove and recreate build task to customize it for Vite
|
|
115
|
+
tsProject.removeTask('build');
|
|
116
|
+
tsProject.addTask('build', {
|
|
117
|
+
description: 'Build for production',
|
|
118
|
+
exec: 'tsc -b && vite build',
|
|
119
|
+
});
|
|
120
|
+
tsProject.addTask('preview', {
|
|
121
|
+
description: 'Preview production build',
|
|
122
|
+
exec: 'vite preview',
|
|
123
|
+
});
|
|
124
|
+
tsProject.addTask('lint', {
|
|
125
|
+
description: 'Lint code',
|
|
126
|
+
exec: 'eslint .',
|
|
127
|
+
});
|
|
128
|
+
}
|
|
129
|
+
}
|
|
130
|
+
/**
|
|
131
|
+
* Creates the ESLint configuration for Vite + React
|
|
132
|
+
*/
|
|
133
|
+
class ViteEslintConfig extends projen_1.FileBase {
|
|
134
|
+
constructor(project) {
|
|
135
|
+
super(project, 'eslint.config.js', {
|
|
136
|
+
marker: true,
|
|
137
|
+
readonly: true,
|
|
138
|
+
});
|
|
139
|
+
}
|
|
140
|
+
synthesizeContent() {
|
|
141
|
+
return `import js from '@eslint/js'
|
|
142
|
+
import globals from 'globals'
|
|
143
|
+
import reactHooks from 'eslint-plugin-react-hooks'
|
|
144
|
+
import reactRefresh from 'eslint-plugin-react-refresh'
|
|
145
|
+
import tseslint from 'typescript-eslint'
|
|
146
|
+
|
|
147
|
+
export default [
|
|
148
|
+
{ ignores: ['dist'] },
|
|
149
|
+
{
|
|
150
|
+
files: ['**/*.{ts,tsx}'],
|
|
151
|
+
languageOptions: {
|
|
152
|
+
ecmaVersion: 2020,
|
|
153
|
+
globals: globals.browser,
|
|
154
|
+
},
|
|
155
|
+
plugins: {
|
|
156
|
+
'react-hooks': reactHooks,
|
|
157
|
+
'react-refresh': reactRefresh,
|
|
158
|
+
},
|
|
159
|
+
rules: {
|
|
160
|
+
...reactHooks.configs.recommended.rules,
|
|
161
|
+
'react-refresh/only-export-components': [
|
|
162
|
+
'warn',
|
|
163
|
+
{ allowConstantExport: true },
|
|
164
|
+
],
|
|
165
|
+
},
|
|
166
|
+
},
|
|
167
|
+
js.configs.recommended,
|
|
168
|
+
...tseslint.configs.recommended,
|
|
169
|
+
]
|
|
170
|
+
`;
|
|
171
|
+
}
|
|
172
|
+
}
|
|
173
|
+
/**
|
|
174
|
+
* Creates the vite.config.ts file
|
|
175
|
+
*/
|
|
176
|
+
class ViteConfigFile extends projen_1.FileBase {
|
|
177
|
+
constructor(project) {
|
|
178
|
+
super(project, 'vite.config.ts', {
|
|
179
|
+
marker: true,
|
|
180
|
+
readonly: true,
|
|
181
|
+
});
|
|
182
|
+
}
|
|
183
|
+
synthesizeContent() {
|
|
184
|
+
return `import { defineConfig } from 'vite'
|
|
185
|
+
import react from '@vitejs/plugin-react'
|
|
186
|
+
|
|
187
|
+
// https://vite.dev/config/
|
|
188
|
+
export default defineConfig({
|
|
189
|
+
plugins: [react()],
|
|
190
|
+
})
|
|
191
|
+
`;
|
|
192
|
+
}
|
|
193
|
+
}
|
|
194
|
+
/**
|
|
195
|
+
* Creates the postcss.config.js file for Tailwind CSS v4
|
|
196
|
+
*/
|
|
197
|
+
class PostCSSConfigFile extends projen_1.FileBase {
|
|
198
|
+
constructor(project) {
|
|
199
|
+
super(project, 'postcss.config.js', {
|
|
200
|
+
marker: true,
|
|
201
|
+
readonly: true,
|
|
202
|
+
});
|
|
203
|
+
}
|
|
204
|
+
synthesizeContent() {
|
|
205
|
+
return `export default {
|
|
206
|
+
plugins: {
|
|
207
|
+
"@tailwindcss/postcss": {},
|
|
208
|
+
autoprefixer: {},
|
|
209
|
+
},
|
|
210
|
+
}
|
|
211
|
+
`;
|
|
212
|
+
}
|
|
213
|
+
}
|
|
214
|
+
/**
|
|
215
|
+
* Creates the vite-env.d.ts file for type declarations
|
|
216
|
+
*/
|
|
217
|
+
class ViteEnvDts extends projen_1.FileBase {
|
|
218
|
+
constructor(project, options) {
|
|
219
|
+
super(project, `${options.srcdir}/vite-env.d.ts`, {
|
|
220
|
+
marker: true,
|
|
221
|
+
readonly: true,
|
|
222
|
+
});
|
|
223
|
+
}
|
|
224
|
+
synthesizeContent() {
|
|
225
|
+
return `/// <reference types="vite/client" />
|
|
226
|
+
`;
|
|
227
|
+
}
|
|
228
|
+
}
|
|
229
|
+
/**
|
|
230
|
+
* Creates the config.ts file for runtime configuration
|
|
231
|
+
*/
|
|
232
|
+
class ConfigFile extends projen_1.FileBase {
|
|
233
|
+
constructor(project, options) {
|
|
234
|
+
super(project, `${options.srcdir}/config.ts`, {
|
|
235
|
+
marker: true,
|
|
236
|
+
readonly: true,
|
|
237
|
+
});
|
|
238
|
+
}
|
|
239
|
+
synthesizeContent() {
|
|
240
|
+
// Get custom config fields from parent project
|
|
241
|
+
const viteProject = this.project;
|
|
242
|
+
const customFields = viteProject.getConfigFields();
|
|
243
|
+
// Generate base config type
|
|
244
|
+
const baseConfigType = `type BaseConfig = {
|
|
245
|
+
clerkPublishableKey: string;
|
|
246
|
+
apiBaseUrl: string;
|
|
247
|
+
timeout: number;
|
|
248
|
+
}`;
|
|
249
|
+
// Generate custom fields type if any exist
|
|
250
|
+
let configTypeDefinition;
|
|
251
|
+
if (customFields.size > 0) {
|
|
252
|
+
const fieldDefinitions = Array.from(customFields.entries())
|
|
253
|
+
.map(([name, type]) => ` ${name}: ${type};`)
|
|
254
|
+
.join('\n');
|
|
255
|
+
const customFieldsType = `type CustomConfig = {
|
|
256
|
+
${fieldDefinitions}
|
|
257
|
+
}`;
|
|
258
|
+
configTypeDefinition = `${customFieldsType}
|
|
259
|
+
|
|
260
|
+
export type Config = BaseConfig & CustomConfig;`;
|
|
261
|
+
}
|
|
262
|
+
else {
|
|
263
|
+
configTypeDefinition = 'export type Config = BaseConfig;';
|
|
264
|
+
}
|
|
265
|
+
return `import { useEffect, useState } from 'react';
|
|
266
|
+
|
|
267
|
+
${baseConfigType}
|
|
268
|
+
|
|
269
|
+
${configTypeDefinition}
|
|
270
|
+
|
|
271
|
+
export function useConfig() {
|
|
272
|
+
const [loadedConfig, setLoadedConfig] = useState<Config|undefined>();
|
|
273
|
+
const [isLoading, setIsLoading] = useState(false);
|
|
274
|
+
|
|
275
|
+
useEffect(() => {
|
|
276
|
+
if (!loadedConfig) {
|
|
277
|
+
fetch('/config.json', { cache: 'no-store' }).then(response=>{
|
|
278
|
+
response.json().then((config) => {
|
|
279
|
+
setIsLoading(false)
|
|
280
|
+
setLoadedConfig(config)
|
|
281
|
+
});
|
|
282
|
+
});
|
|
283
|
+
|
|
284
|
+
}
|
|
285
|
+
}, [loadedConfig]);
|
|
286
|
+
|
|
287
|
+
return { config: loadedConfig, isLoading };
|
|
288
|
+
}
|
|
289
|
+
`;
|
|
290
|
+
}
|
|
291
|
+
}
|
|
292
|
+
/**
|
|
293
|
+
* Creates the config.json sample file for runtime configuration
|
|
294
|
+
*/
|
|
295
|
+
class ConfigJsonFile extends projen_1.SampleFile {
|
|
296
|
+
constructor(project, _options) {
|
|
297
|
+
super(project, `${project.publicDir}/config.json`, {
|
|
298
|
+
contents: JSON.stringify({
|
|
299
|
+
apiBaseUrl: 'http://localhost:3000/api',
|
|
300
|
+
clerkPublishableKey: 'tbd',
|
|
301
|
+
}, null, 2),
|
|
302
|
+
});
|
|
303
|
+
this.path = `${project.publicDir}/config.json`;
|
|
304
|
+
}
|
|
305
|
+
}
|
|
306
|
+
/**
|
|
307
|
+
* Creates the index.html file
|
|
308
|
+
*/
|
|
309
|
+
class IndexHtml extends projen_1.FileBase {
|
|
310
|
+
constructor(project, options) {
|
|
311
|
+
super(project, 'index.html', {
|
|
312
|
+
marker: true,
|
|
313
|
+
readonly: true,
|
|
314
|
+
});
|
|
315
|
+
this.title = options.title;
|
|
316
|
+
this.srcdir = options.srcdir;
|
|
317
|
+
}
|
|
318
|
+
synthesizeContent() {
|
|
319
|
+
return `<!doctype html>
|
|
320
|
+
<html lang="en">
|
|
321
|
+
<head>
|
|
322
|
+
<meta charset="UTF-8" />
|
|
323
|
+
<link rel="icon" type="image/svg+xml" href="/vite.svg" />
|
|
324
|
+
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
|
325
|
+
<title>${this.title}</title>
|
|
326
|
+
</head>
|
|
327
|
+
<body>
|
|
328
|
+
<div id="root"></div>
|
|
329
|
+
<script type="module" src="/${this.srcdir}/main.tsx"></script>
|
|
330
|
+
</body>
|
|
331
|
+
</html>
|
|
332
|
+
`;
|
|
333
|
+
}
|
|
334
|
+
}
|
|
335
|
+
/**
|
|
336
|
+
* Creates the TypeScript configuration files for Vite
|
|
337
|
+
*/
|
|
338
|
+
class ViteTsConfigs extends projen_1.Component {
|
|
339
|
+
constructor(project, options) {
|
|
340
|
+
super(project);
|
|
341
|
+
// Main tsconfig.json (project references)
|
|
342
|
+
new projen_1.JsonFile(project, 'tsconfig.json', {
|
|
343
|
+
marker: true,
|
|
344
|
+
readonly: true,
|
|
345
|
+
obj: {
|
|
346
|
+
files: [],
|
|
347
|
+
references: [
|
|
348
|
+
{ path: './tsconfig.app.json' },
|
|
349
|
+
{ path: './tsconfig.node.json' },
|
|
350
|
+
],
|
|
351
|
+
},
|
|
352
|
+
});
|
|
353
|
+
// tsconfig.app.json (for application code)
|
|
354
|
+
new projen_1.JsonFile(project, 'tsconfig.app.json', {
|
|
355
|
+
marker: true,
|
|
356
|
+
readonly: true,
|
|
357
|
+
obj: {
|
|
358
|
+
compilerOptions: {
|
|
359
|
+
target: 'ES2020',
|
|
360
|
+
useDefineForClassFields: true,
|
|
361
|
+
lib: ['ES2020', 'DOM', 'DOM.Iterable'],
|
|
362
|
+
module: 'ESNext',
|
|
363
|
+
skipLibCheck: true,
|
|
364
|
+
// Bundler mode
|
|
365
|
+
moduleResolution: 'bundler',
|
|
366
|
+
allowImportingTsExtensions: true,
|
|
367
|
+
isolatedModules: true,
|
|
368
|
+
moduleDetection: 'force',
|
|
369
|
+
noEmit: true,
|
|
370
|
+
jsx: 'react-jsx',
|
|
371
|
+
// Linting
|
|
372
|
+
strict: true,
|
|
373
|
+
noUnusedLocals: true,
|
|
374
|
+
noUnusedParameters: true,
|
|
375
|
+
noFallthroughCasesInSwitch: true,
|
|
376
|
+
noUncheckedSideEffectImports: true,
|
|
377
|
+
},
|
|
378
|
+
include: [options.srcdir],
|
|
379
|
+
},
|
|
380
|
+
});
|
|
381
|
+
// tsconfig.node.json (for Vite config)
|
|
382
|
+
new projen_1.JsonFile(project, 'tsconfig.node.json', {
|
|
383
|
+
marker: true,
|
|
384
|
+
readonly: true,
|
|
385
|
+
obj: {
|
|
386
|
+
compilerOptions: {
|
|
387
|
+
target: 'ES2022',
|
|
388
|
+
lib: ['ES2023'],
|
|
389
|
+
module: 'ESNext',
|
|
390
|
+
skipLibCheck: true,
|
|
391
|
+
// Bundler mode
|
|
392
|
+
moduleResolution: 'bundler',
|
|
393
|
+
allowImportingTsExtensions: true,
|
|
394
|
+
isolatedModules: true,
|
|
395
|
+
moduleDetection: 'force',
|
|
396
|
+
noEmit: true,
|
|
397
|
+
// Linting
|
|
398
|
+
strict: true,
|
|
399
|
+
noUnusedLocals: true,
|
|
400
|
+
noUnusedParameters: true,
|
|
401
|
+
noFallthroughCasesInSwitch: true,
|
|
402
|
+
},
|
|
403
|
+
include: ['vite.config.ts'],
|
|
404
|
+
},
|
|
405
|
+
});
|
|
406
|
+
}
|
|
407
|
+
}
|
|
408
|
+
/**
|
|
409
|
+
* Generates sample code for a Vite + React project
|
|
410
|
+
*/
|
|
411
|
+
class ViteSampleCode extends projen_1.Component {
|
|
412
|
+
constructor(project, options) {
|
|
413
|
+
super(project);
|
|
414
|
+
// Create App.tsx
|
|
415
|
+
new projen_1.SampleFile(project, `${options.srcdir}/App.tsx`, {
|
|
416
|
+
contents: `import './App.css'
|
|
417
|
+
import { SignedIn, UserButton } from '@clerk/clerk-react';
|
|
418
|
+
import { useBaseQuery } from "./services/BaseService.ts";
|
|
419
|
+
|
|
420
|
+
|
|
421
|
+
function App() {
|
|
422
|
+
const {data, isLoading} = useBaseQuery('/', {retryOnMount: true});
|
|
423
|
+
|
|
424
|
+
|
|
425
|
+
return (
|
|
426
|
+
<>
|
|
427
|
+
<header>
|
|
428
|
+
|
|
429
|
+
<SignedIn>
|
|
430
|
+
<UserButton/>
|
|
431
|
+
</SignedIn>
|
|
432
|
+
</header>
|
|
433
|
+
<div className="card">
|
|
434
|
+
{ isLoading ? 'Loading...' : JSON.stringify(data, null ,)}
|
|
435
|
+
</div>
|
|
436
|
+
</>
|
|
437
|
+
)
|
|
438
|
+
}
|
|
439
|
+
|
|
440
|
+
export default App
|
|
441
|
+
`,
|
|
442
|
+
});
|
|
443
|
+
// Create BaseService.ts
|
|
444
|
+
new projen_1.TextFile(project, `${options.srcdir}/services/BaseService.ts`, {
|
|
445
|
+
marker: true,
|
|
446
|
+
readonly: true,
|
|
447
|
+
lines: [
|
|
448
|
+
'import { useQuery, UseQueryOptions } from \'@tanstack/react-query\';',
|
|
449
|
+
'import { useAuth } from \'@clerk/clerk-react\';',
|
|
450
|
+
'import { useConfig } from \'../config\';',
|
|
451
|
+
'',
|
|
452
|
+
'export function useBaseQuery<TData = unknown>(',
|
|
453
|
+
' path: string,',
|
|
454
|
+
' options?: Omit<UseQueryOptions<TData>, \'queryKey\' | \'queryFn\'>',
|
|
455
|
+
') {',
|
|
456
|
+
' const { config } = useConfig();',
|
|
457
|
+
' const { getToken } = useAuth();',
|
|
458
|
+
'',
|
|
459
|
+
' return useQuery<TData>({',
|
|
460
|
+
' queryKey: [\'base\'],',
|
|
461
|
+
' queryFn: async () => {',
|
|
462
|
+
' if (!config?.apiBaseUrl) {',
|
|
463
|
+
' throw new Error(\'API base URL not configured\');',
|
|
464
|
+
' }',
|
|
465
|
+
'',
|
|
466
|
+
' const token = await getToken();',
|
|
467
|
+
'',
|
|
468
|
+
' // Normalize URL by removing trailing slash from baseUrl and ensuring path starts with slash',
|
|
469
|
+
' const baseUrl = config.apiBaseUrl.replace(/\\/+$/, \'\');',
|
|
470
|
+
' const normalizedPath = path.startsWith(\'/\') ? path : `/${path}`;',
|
|
471
|
+
' ',
|
|
472
|
+
' const response = await fetch(`${baseUrl}${normalizedPath}`, {',
|
|
473
|
+
' headers: {',
|
|
474
|
+
' \'Authorization\': `Bearer ${token}`,',
|
|
475
|
+
' },',
|
|
476
|
+
' });',
|
|
477
|
+
'',
|
|
478
|
+
' if (!response.ok) {',
|
|
479
|
+
' throw new Error(`HTTP error! status: ${response.status}`);',
|
|
480
|
+
' }',
|
|
481
|
+
'',
|
|
482
|
+
' return response.json();',
|
|
483
|
+
' },',
|
|
484
|
+
' enabled: !!config?.apiBaseUrl,',
|
|
485
|
+
' ...options,',
|
|
486
|
+
' });',
|
|
487
|
+
'}',
|
|
488
|
+
],
|
|
489
|
+
});
|
|
490
|
+
// Create RootService.ts
|
|
491
|
+
new projen_1.SampleFile(project, `${options.srcdir}/services/RootService.ts`, {
|
|
492
|
+
contents: `import { UseQueryOptions } from '@tanstack/react-query';
|
|
493
|
+
import { useBaseQuery } from './BaseService';
|
|
494
|
+
|
|
495
|
+
export function useRootQuery<TData = unknown>(
|
|
496
|
+
options?: Omit<UseQueryOptions<TData>, 'queryKey' | 'queryFn'>
|
|
497
|
+
) {
|
|
498
|
+
return useBaseQuery<TData>('/', options);
|
|
499
|
+
}
|
|
500
|
+
`,
|
|
501
|
+
});
|
|
502
|
+
// Create App.css
|
|
503
|
+
new projen_1.SampleFile(project, `${options.srcdir}/App.css`, {
|
|
504
|
+
contents: `#root {
|
|
505
|
+
width: 100%;
|
|
506
|
+
height: 100%;
|
|
507
|
+
}
|
|
508
|
+
|
|
509
|
+
.logo {
|
|
510
|
+
height: 6em;
|
|
511
|
+
padding: 1.5em;
|
|
512
|
+
will-change: filter;
|
|
513
|
+
transition: filter 300ms;
|
|
514
|
+
}
|
|
515
|
+
.logo:hover {
|
|
516
|
+
filter: drop-shadow(0 0 2em #646cffaa);
|
|
517
|
+
}
|
|
518
|
+
.logo.react:hover {
|
|
519
|
+
filter: drop-shadow(0 0 2em #61dafbaa);
|
|
520
|
+
}
|
|
521
|
+
|
|
522
|
+
@keyframes logo-spin {
|
|
523
|
+
from {
|
|
524
|
+
transform: rotate(0deg);
|
|
525
|
+
}
|
|
526
|
+
to {
|
|
527
|
+
transform: rotate(360deg);
|
|
528
|
+
}
|
|
529
|
+
}
|
|
530
|
+
|
|
531
|
+
@media (prefers-reduced-motion: no-preference) {
|
|
532
|
+
a:nth-of-type(2) .logo {
|
|
533
|
+
animation: logo-spin infinite 20s linear;
|
|
534
|
+
}
|
|
535
|
+
}
|
|
536
|
+
|
|
537
|
+
.card {
|
|
538
|
+
padding: 2em;
|
|
539
|
+
}
|
|
540
|
+
|
|
541
|
+
.read-the-docs {
|
|
542
|
+
color: #888;
|
|
543
|
+
}
|
|
544
|
+
`,
|
|
545
|
+
});
|
|
546
|
+
// Create main.tsx
|
|
547
|
+
new projen_1.SampleFile(project, `${options.srcdir}/main.tsx`, {
|
|
548
|
+
contents: `import { StrictMode } from 'react'
|
|
549
|
+
import { createRoot } from 'react-dom/client'
|
|
550
|
+
import './index.css'
|
|
551
|
+
import App from './App.tsx'
|
|
552
|
+
import { ClerkProvider, SignedIn, SignedOut, SignInButton } from '@clerk/clerk-react'
|
|
553
|
+
import { useConfig } from './config'
|
|
554
|
+
import { QueryClient, QueryClientProvider } from "@tanstack/react-query"
|
|
555
|
+
|
|
556
|
+
function AppWrapper() {
|
|
557
|
+
const { config, isLoading } = useConfig();
|
|
558
|
+
const queryClient = new QueryClient()
|
|
559
|
+
if (isLoading || !config) {
|
|
560
|
+
return (
|
|
561
|
+
<div className="loading-screen">
|
|
562
|
+
Loading...
|
|
563
|
+
</div>
|
|
564
|
+
);
|
|
565
|
+
}
|
|
566
|
+
|
|
567
|
+
return (
|
|
568
|
+
<ClerkProvider publishableKey={config.clerkPublishableKey}>
|
|
569
|
+
<QueryClientProvider client={queryClient}>
|
|
570
|
+
<div className="w-full h-full">
|
|
571
|
+
<SignedOut>
|
|
572
|
+
<div className="flex items-center justify-center min-h-screen">
|
|
573
|
+
<SignInButton/>
|
|
574
|
+
</div>
|
|
575
|
+
</SignedOut>
|
|
576
|
+
<SignedIn>
|
|
577
|
+
<App/>
|
|
578
|
+
</SignedIn>
|
|
579
|
+
</div>
|
|
580
|
+
</QueryClientProvider>
|
|
581
|
+
</ClerkProvider>
|
|
582
|
+
);
|
|
583
|
+
}
|
|
584
|
+
|
|
585
|
+
createRoot(document.getElementById('root')!).render(
|
|
586
|
+
<StrictMode>
|
|
587
|
+
<AppWrapper/>
|
|
588
|
+
</StrictMode>
|
|
589
|
+
)
|
|
590
|
+
`,
|
|
591
|
+
});
|
|
592
|
+
// Create index.css
|
|
593
|
+
new projen_1.SampleFile(project, `${options.srcdir}/index.css`, {
|
|
594
|
+
contents: `@import "tailwindcss";
|
|
595
|
+
|
|
596
|
+
@theme {
|
|
597
|
+
--color-scheme: light dark;
|
|
598
|
+
}
|
|
599
|
+
|
|
600
|
+
:root {
|
|
601
|
+
font-family: Inter, system-ui, Avenir, Helvetica, Arial, sans-serif;
|
|
602
|
+
line-height: 1.5;
|
|
603
|
+
font-weight: 400;
|
|
604
|
+
|
|
605
|
+
color-scheme: light dark;
|
|
606
|
+
|
|
607
|
+
font-synthesis: none;
|
|
608
|
+
text-rendering: optimizeLegibility;
|
|
609
|
+
-webkit-font-smoothing: antialiased;
|
|
610
|
+
-moz-osx-font-smoothing: grayscale;
|
|
611
|
+
}
|
|
612
|
+
|
|
613
|
+
a {
|
|
614
|
+
font-weight: 500;
|
|
615
|
+
color: #646cff;
|
|
616
|
+
text-decoration: inherit;
|
|
617
|
+
}
|
|
618
|
+
a:hover {
|
|
619
|
+
color: #535bf2;
|
|
620
|
+
}
|
|
621
|
+
|
|
622
|
+
body {
|
|
623
|
+
margin: 0;
|
|
624
|
+
min-width: 320px;
|
|
625
|
+
min-height: 100vh;
|
|
626
|
+
}
|
|
627
|
+
|
|
628
|
+
h1 {
|
|
629
|
+
font-size: 3.2em;
|
|
630
|
+
line-height: 1.1;
|
|
631
|
+
}
|
|
632
|
+
|
|
633
|
+
button {
|
|
634
|
+
font-family: inherit;
|
|
635
|
+
cursor: pointer;
|
|
636
|
+
}
|
|
637
|
+
|
|
638
|
+
.loading-screen {
|
|
639
|
+
display: flex;
|
|
640
|
+
justify-content: center;
|
|
641
|
+
align-items: center;
|
|
642
|
+
height: 100vh;
|
|
643
|
+
font-size: 1.5rem;
|
|
644
|
+
}
|
|
645
|
+
|
|
646
|
+
html, body, #root {
|
|
647
|
+
width: 100%;
|
|
648
|
+
height: 100%;
|
|
649
|
+
}
|
|
650
|
+
`,
|
|
651
|
+
});
|
|
652
|
+
}
|
|
653
|
+
}
|
|
654
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiVml0ZVJlYWN0UHJvamVjdC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uL3NyYy9WaXRlUmVhY3RQcm9qZWN0LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUFBLCtCQUEwQztBQUMxQyxtQ0FBc0Y7QUFDdEYsc0RBQXVGO0FBNEJ2Rjs7R0FFRztBQUNILE1BQWEsZ0JBQWlCLFNBQVEsaUNBQW9CO0lBTXhELFlBQVksT0FBZ0M7UUFDMUMsTUFBTSxFQUNKLFVBQVUsR0FBRyxJQUFJLEVBQ2pCLE1BQU0sR0FBRyxLQUFLLEVBQ2QsU0FBUyxHQUFHLFFBQVEsRUFDcEIsS0FBSyxFQUNMLEdBQUcsV0FBVyxFQUNmLEdBQUcsT0FBTyxDQUFDO1FBQ1osS0FBSyxDQUFDO1lBQ0osR0FBRyxXQUFXO1lBQ2QsVUFBVSxFQUFFLEtBQUssRUFBRSxxQ0FBcUM7WUFDeEQsTUFBTSxFQUFFLEtBQUssRUFBRSxtQ0FBbUM7WUFDbEQsSUFBSSxFQUFFLEtBQUssRUFBRSw2QkFBNkI7WUFDMUMsT0FBTyxFQUFFLElBQUk7WUFDYixlQUFlLEVBQUUsSUFBSSxFQUFFLGtDQUFrQztZQUN6RCxJQUFJLEVBQUU7Z0JBQ0osb0JBQW9CO2dCQUNwQix1QkFBdUI7Z0JBQ3ZCLHNCQUFzQjtnQkFDdEIsY0FBYztnQkFDZCxTQUFTO2dCQUNULGFBQWE7YUFDZDtTQUNGLENBQUMsQ0FBQztRQUVILG1DQUFtQztRQUNuQyxJQUFJLENBQUMsWUFBWSxHQUFHLElBQUksR0FBRyxFQUFrQixDQUFDO1FBRTlDLElBQUksQ0FBQyxNQUFNLEdBQUcsTUFBTSxDQUFDO1FBQ3JCLElBQUksQ0FBQyxTQUFTLEdBQUcsU0FBUyxDQUFDO1FBRTNCLHFCQUFxQjtRQUNyQixJQUFJLGFBQWEsQ0FBQyxJQUFJLEVBQUU7WUFDdEIsTUFBTSxFQUFFLElBQUksQ0FBQyxNQUFNO1NBQ3BCLENBQUMsQ0FBQztRQUVILDJCQUEyQjtRQUMzQixJQUFJLGdCQUFnQixDQUFDLElBQUksQ0FBQyxD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Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztDQXdDZjtTQUNJLENBQUMsQ0FBQztRQUVILGtCQUFrQjtRQUNsQixJQUFJLG1CQUFVLENBQUMsT0FBTyxFQUFFLEdBQUcsT0FBTyxDQUFDLE1BQU0sV0FBVyxFQUFFO1lBQ3BELFFBQVEsRUFBRTs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0NBMENmO1NBQ0ksQ0FBQyxDQUFDO1FBRUgsbUJBQW1CO1FBQ25CLElBQUksbUJBQVUsQ0FBQyxPQUFPLEVBQUUsR0FBRyxPQUFPLENBQUMsTUFBTSxZQUFZLEVBQUU7WUFDckQsUUFBUSxFQUFFOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztDQXdEZjtTQUNJLENBQUMsQ0FBQztJQUVMLENBQUM7Q0FDRiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IHJlbGF0aXZlLCBiYXNlbmFtZSB9IGZyb20gJ3BhdGgnO1xuaW1wb3J0IHsgQ29tcG9uZW50LCBGaWxlQmFzZSwgSnNvbkZpbGUsIFByb2plY3QsIFNhbXBsZUZpbGUsIFRleHRGaWxlIH0gZnJvbSAncHJvamVuJztcbmltcG9ydCB7IFR5cGVTY3JpcHRBcHBQcm9qZWN0LCBUeXBlU2NyaXB0UHJvamVjdE9wdGlvbnMgfSBmcm9tICdwcm9qZW4vbGliL3R5cGVzY3JpcHQnO1xuXG5leHBvcnQgaW50ZXJmYWNlIFZpdGVSZWFjdFByb2plY3RPcHRpb25zIGV4dGVuZHMgVHlwZVNjcmlwdFByb2plY3RPcHRpb25zIHtcbiAgLyoqXG4gICAqIFdoZXRoZXIgdG8gZ2VuZXJhdGUgc2FtcGxlIGNvZGVcbiAgICogQGRlZmF1bHQgdHJ1ZVxuICAgKi9cbiAgcmVhZG9ubHkgc2FtcGxlQ29kZT86IGJvb2xlYW47XG5cbiAgLyoqXG4gICAqIFNvdXJjZSBkaXJlY3RvcnlcbiAgICogQGRlZmF1bHQgXCJzcmNcIlxuICAgKi9cbiAgcmVhZG9ubHkgc3JjZGlyPzogc3RyaW5nO1xuXG4gIC8qKlxuICAgKiBQdWJsaWMgZGlyZWN0b3J5IGZvciBzdGF0aWMgYXNzZXRzXG4gICAqIEBkZWZhdWx0IFwicHVibGljXCJcbiAgICovXG4gIHJlYWRvbmx5IHB1YmxpY0Rpcj86IHN0cmluZztcblxuICAvKipcbiAgICogVGl0bGUgZm9yIHRoZSBIVE1MIHBhZ2VcbiAgICogQGRlZmF1bHQgcHJvamVjdCBuYW1lXG4gICAqL1xuICByZWFkb25seSB0aXRsZT86IHN0cmluZztcbn1cblxuLyoqXG4gKiBWaXRlICsgUmVhY3QgVHlwZVNjcmlwdCBwcm9qZWN0XG4gKi9cbmV4cG9ydCBjbGFzcyBWaXRlUmVhY3RQcm9qZWN0IGV4dGVuZHMgVHlwZVNjcmlwdEFwcFByb2plY3Qge1xuICBwdWJsaWMgcmVhZG9ubHkgc3JjZGlyOiBzdHJpbmc7XG4gIHB1YmxpYyByZWFkb25seSBwdWJsaWNEaXI6IHN0cmluZztcbiAgcHVibGljIHJlYWRvbmx5IGRldkNvbW1hbmQ6IHN0cmluZztcbiAgcHJpdmF0ZSByZWFkb25seSBjb25maWdGaWVsZHM6IE1hcDxzdHJpbmcsIHN0cmluZz47XG5cbiAgY29uc3RydWN0b3Iob3B0aW9uczogVml0ZVJlYWN0UHJvamVjdE9wdGlvbnMpIHtcbiAgICBjb25zdCB7XG4gICAgICBzYW1wbGVDb2RlID0gdHJ1ZSxcbiAgICAgIHNyY2RpciA9ICdzcmMnLFxuICAgICAgcHVibGljRGlyID0gJ3B1YmxpYycsXG4gICAgICB0aXRsZSxcbiAgICAgIC4uLnJlc3RPcHRpb25zXG4gICAgfSA9IG9wdGlvbnM7XG4gICAgc3VwZXIoe1xuICAgICAgLi4ucmVzdE9wdGlvbnMsXG4gICAgICBzYW1wbGVDb2RlOiBmYWxzZSwgLy8gV2UnbGwgaGFuZGxlIHNhbXBsZSBjb2RlIG91cnNlbHZlc1xuICAgICAgZXNsaW50OiBmYWxzZSwgLy8gV2UnbGwgY29uZmlndXJlIGVzbGludCBvdXJzZWx2ZXNcbiAgICAgIGplc3Q6IGZhbHNlLCAvLyBWaXRlIHVzZXMgdml0ZXN0IHR5cGljYWxseVxuICAgICAgcGFja2FnZTogdHJ1ZSxcbiAgICAgIGRpc2FibGVUc2NvbmZpZzogdHJ1ZSwgLy8gV2UnbGwgbWFuYWdlIHRzY29uZmlnIG91cnNlbHZlc1xuICAgICAgZGVwczogW1xuICAgICAgICAnQGNsZXJrL2NsZXJrLXJlYWN0JyxcbiAgICAgICAgJ0B0YW5zdGFjay9yZWFjdC1xdWVyeScsXG4gICAgICAgICdAdGFpbHdpbmRjc3MvcG9zdGNzcycsXG4gICAgICAgICdhdXRvcHJlZml4ZXInLFxuICAgICAgICAncG9zdGNzcycsXG4gICAgICAgICd0YWlsd2luZGNzcycsXG4gICAgICBdLFxuICAgIH0pO1xuXG4gICAgLy8gSW5pdGlhbGl6ZSBjb25maWcgZmllbGRzIHN0b3JhZ2VcbiAgICB0aGlzLmNvbmZpZ0ZpZWxkcyA9IG5ldyBNYXA8c3RyaW5nLCBzdHJpbmc+KCk7XG5cbiAgICB0aGlzLnNyY2RpciA9IHNyY2RpcjtcbiAgICB0aGlzLnB1YmxpY0RpciA9IHB1YmxpY0RpcjtcblxuICAgIC8vIEFkZCBWaXRlIGNvbXBvbmVudFxuICAgIG5ldyBWaXRlQ29tcG9uZW50KHRoaXMsIHtcbiAgICAgIHNyY2RpcjogdGhpcy5zcmNkaXIsXG4gICAgfSk7XG5cbiAgICAvLyBBZGQgRVNMaW50IGNvbmZpZ3VyYXRpb25cbiAgICBuZXcgVml0ZUVzbGludENvbmZpZyh0aGlzKTtcblxuICAgIC8vIEFkZCBpbmRleC5odG1sXG4gICAgbmV3IEluZGV4SHRtbCh0aGlzLCB7XG4gICAgICB0aXRsZTogdGl0bGUgPz8gdGhpcy5uYW1lLFxuICAgICAgc3JjZGlyOiB0aGlzLnNyY2RpcixcbiAgICB9KTtcblxuICAgIC8vIEFkZCBUeXBlU2NyaXB0IGNvbmZpZ3NcbiAgICBuZXcgVml0ZVRzQ29uZmlncyh0aGlzLCB7XG4gICAgICBzcmNkaXI6IHRoaXMuc3JjZGlyLFxuICAgIH0pO1xuXG4gICAgLy8gQWRkIC5naXRpZ25vcmUgZW50cmllc1xuICAgIHRoaXMuZ2l0aWdub3JlPy5hZGRQYXR0ZXJucygnZGlzdCcsICdkaXN0LXNzcicsICcqLmxvY2FsJyk7XG5cbiAgICAvLyBBZGQgdml0ZS1lbnYuZC50cyBmb3IgdHlwZSBkZWNsYXJhdGlvbnNcbiAgICBuZXcgVml0ZUVudkR0cyh0aGlzLCB7XG4gICAgICBzcmNkaXI6IHRoaXMuc3JjZGlyLFxuICAgIH0pO1xuXG4gICAgLy8gQWRkIFBvc3RDU1MgY29uZmlnIGZvciBUYWlsd2luZFxuICAgIG5ldyBQb3N0Q1NTQ29uZmlnRmlsZSh0aGlzKTtcblxuICAgIC8vIEFkZCBjb25maWcudHMgZm9yIHJ1bnRpbWUgY29uZmlndXJhdGlvblxuICAgIG5ldyBDb25maWdGaWxlKHRoaXMsIHtcbiAgICAgIHNyY2RpcjogdGhpcy5zcmNkaXIsXG4gICAgfSk7XG5cbiAgICAvLyBBZGQgc2FtcGxlIGNvbmZpZy5qc29uXG4gICAgbmV3IENvbmZpZ0pzb25GaWxlKHRoaXMsIHsgfSk7XG5cblxuICAgIC8vIEFkZCBzYW1wbGUgY29kZSBpZiByZXF1ZXN0ZWRcbiAgICBpZiAoc2FtcGxlQ29kZSkge1xuICAgICAgbmV3IFZpdGVTYW1wbGVDb2RlKHRoaXMsIHtcbiAgICAgICAgc3JjZGlyOiB0aGlzLnNyY2RpcixcbiAgICAgICAgcHVibGljRGlyOiB0aGlzLnB1YmxpY0RpcixcbiAgICAgIH0pO1xuICAgIH1cblxuICAgIC8vIFVzZSByZWxhdGl2ZSBwYXRoIGZvciBkZXZDb21tYW5kXG4gICAgY29uc3QgcmVsYXRpdmVPdXRkaXIgPSB0aGlzLnBhcmVudFxuICAgICAgPyByZWxhdGl2ZSh0aGlzLnBhcmVudC5vdXRkaXIsIHRoaXMub3V0ZGlyKVxuICAgICAgOiBiYXNlbmFtZSh0aGlzLm91dGRpcik7XG4gICAgdGhpcy5kZXZDb21tYW5kID0gYGNkICR7cmVsYXRpdmVPdXRkaXJ9ICYmIHlhcm4gZGV2YDtcbiAgfVxuXG4gIC8qKlxuICAgKiBBZGQgYSBjdXN0b20gZmllbGQgdG8gdGhlIENvbmZpZyB0eXBlXG4gICAqIEBwYXJhbSBmaWVsZE5hbWUgVGhlIG5hbWUgb2YgdGhlIGZpZWxkXG4gICAqIEBwYXJhbSBmaWVsZFR5cGUgVGhlIFR5cGVTY3JpcHQgdHlwZSBvZiB0aGUgZmllbGQgKGUuZy4sICdzdHJpbmcnLCAnbnVtYmVyJywgJ2Jvb2xlYW4nLCAnc3RyaW5nW10nKVxuICAgKiBAcmV0dXJucyB0aGlzIHByb2plY3QgaW5zdGFuY2UgZm9yIGNoYWluaW5nXG4gICAqL1xuICBhZGRDb25maWdGaWVsZChmaWVsZE5hbWU6IHN0cmluZywgZmllbGRUeXBlOiBzdHJpbmcpOiB0aGlzIHtcbiAgICB0aGlzLmNvbmZpZ0ZpZWxkcy5zZXQoZmllbGROYW1lLCBmaWVsZFR5cGUpO1xuICAgIHJldHVybiB0aGlzO1xuICB9XG5cbiAgLyoqXG4gICAqIEdldCB0aGUgY29uZmlnIGZpZWxkcyBtYXAgKGZvciBpbnRlcm5hbCB1c2UgYnkgQ29uZmlnRmlsZSlcbiAgICogQGludGVybmFsXG4gICAqL1xuICBnZXRDb25maWdGaWVsZHMoKTogTWFwPHN0cmluZywgc3RyaW5nPiB7XG4gICAgcmV0dXJuIHRoaXMuY29uZmlnRmllbGRzO1xuICB9XG59XG5cbmludGVyZmFjZSBWaXRlQ29tcG9uZW50T3B0aW9ucyB7XG4gIHJlYWRvbmx5IHNyY2Rpcjogc3RyaW5nO1xufVxuXG4vKipcbiAqIENvbmZpZ3VyZXMgVml0ZSBkZXBlbmRlbmNpZXMgYW5kIHNjcmlwdHNcbiAqL1xuY2xhc3MgVml0ZUNvbXBvbmVudCBleHRlbmRzIENvbXBvbmVudCB7XG4gIGNvbnN0cnVjdG9yKHByb2plY3Q6IFR5cGVTY3JpcHRBcHBQcm9qZWN0LCBfb3B0aW9uczogVml0ZUNvbXBvbmVudE9wdGlvbnMpIHtcbiAgICBzdXBlcihwcm9qZWN0KTtcblxuICAgIGNvbnN0IHRzUHJvamVjdCA9IHByb2plY3QgYXMgVHlwZVNjcmlwdEFwcFByb2plY3Q7XG5cbiAgICAvLyBTZXQgcGFja2FnZSB0eXBlIHRvIG1vZHVsZSBmb3IgRVNNIHN1cHBvcnRcbiAgICB0c1Byb2plY3QucGFja2FnZS5hZGRGaWVsZCgndHlwZScsICdtb2R1bGUnKTtcblxuICAgIC8vIEFkZCBSZWFjdCBkZXBlbmRlbmNpZXNcbiAgICB0c1Byb2plY3QuYWRkRGVwcygncmVhY3RAXjE5LjIuMCcsICdyZWFjdC1kb21AXjE5LjIuMCcpO1xuXG4gICAgLy8gQWRkIGRldiBkZXBlbmRlbmNpZXNcbiAgICB0c1Byb2plY3QuYWRkRGV2RGVwcyhcbiAgICAgICdAdHlwZXMvcmVhY3RAXjE5LjIuMicsXG4gICAgICAnQHR5cGVzL3JlYWN0LWRvbUBeMTkuMi4yJyxcbiAgICAgICdAdml0ZWpzL3BsdWdpbi1yZWFjdEBeNS4xLjAnLFxuICAgICAgJ3ZpdGVAXjUuNC4wJyxcbiAgICAgICd0eXBlc2NyaXB0QH41LjkuMycsXG4gICAgICAnQGVzbGludC9qc0BeOS4zOS4xJyxcbiAgICAgICdlc2xpbnRAXjkuMzkuMScsXG4gICAgICAnZXNsaW50LXBsdWdpbi1yZWFjdC1ob29rc0BeNS4yLjAnLFxuICAgICAgJ2VzbGludC1wbHVnaW4tcmVhY3QtcmVmcmVzaEBeMC40LjI0JyxcbiAgICAgICdnbG9iYWxzQF4xNi41LjAnLFxuICAgICAgJ3R5cGVzY3JpcHQtZXNsaW50QF44LjQ2LjMnLFxuICAgICk7XG5cbiAgICAvLyBDcmVhdGUgdml0ZS5jb25maWcudHMgYXMgYSBtYW5hZ2VkIGZpbGVcbiAgICBuZXcgVml0ZUNvbmZpZ0ZpbGUocHJvamVjdCk7XG5cbiAgICAvLyBBZGQgc2NyaXB0c1xuICAgIHRzUHJvamVjdC5hZGRUYXNrKCdkZXYnLCB7XG4gICAgICBkZXNjcmlwdGlvbjogJ1N0YXJ0IGRldmVsb3BtZW50IHNlcnZlcicsXG4gICAgICBleGVjOiAndml0ZScsXG4gICAgfSk7XG5cbiAgICAvLyBSZW1vdmUgYW5kIHJlY3JlYXRlIGJ1aWxkIHRhc2sgdG8gY3VzdG9taXplIGl0IGZvciBWaXRlXG4gICAgdHNQcm9qZWN0LnJlbW92ZVRhc2soJ2J1aWxkJyk7XG4gICAgdHNQcm9qZWN0LmFkZFRhc2soJ2J1aWxkJywge1xuICAgICAgZGVzY3JpcHRpb246ICdCdWlsZCBmb3IgcHJvZHVjdGlvbicsXG4gICAgICBleGVjOiAndHNjIC1iICYmIHZpdGUgYnVpbGQnLFxuICAgIH0pO1xuXG4gICAgdHNQcm9qZWN0LmFkZFRhc2soJ3ByZXZpZXcnLCB7XG4gICAgICBkZXNjcmlwdGlvbjogJ1ByZXZpZXcgcHJvZHVjdGlvbiBidWlsZCcsXG4gICAgICBleGVjOiAndml0ZSBwcmV2aWV3JyxcbiAgICB9KTtcblxuICAgIHRzUHJvamVjdC5hZGRUYXNrKCdsaW50Jywge1xuICAgICAgZGVzY3JpcHRpb246ICdMaW50IGNvZGUnLFxuICAgICAgZXhlYzogJ2VzbGludCAuJyxcbiAgICB9KTtcbiAgfVxufVxuXG4vKipcbiAqIENyZWF0ZXMgdGhlIEVTTGludCBjb25maWd1cmF0aW9uIGZvciBWaXRlICsgUmVhY3RcbiAqL1xuY2xhc3MgVml0ZUVzbGludENvbmZpZyBleHRlbmRzIEZpbGVCYXNlIHtcbiAgY29uc3RydWN0b3IocHJvamVjdDogUHJvamVjdCkge1xuICAgIHN1cGVyKHByb2plY3QsICdlc2xpbnQuY29uZmlnLmpzJywge1xuICAgICAgbWFya2VyOiB0cnVlLFxuICAgICAgcmVhZG9ubHk6IHRydWUsXG4gICAgfSk7XG4gIH1cblxuICBzeW50aGVzaXplQ29udGVudCgpOiBzdHJpbmcgfCB1bmRlZmluZWQge1xuICAgIHJldHVybiBgaW1wb3J0IGpzIGZyb20gJ0Blc2xpbnQvanMnXG5pbXBvcnQgZ2xvYmFscyBmcm9tICdnbG9iYWxzJ1xuaW1wb3J0IHJlYWN0SG9va3MgZnJvbSAnZXNsaW50LXBsdWdpbi1yZWFjdC1ob29rcydcbmltcG9ydCByZWFjdFJlZnJlc2ggZnJvbSAnZXNsaW50LXBsdWdpbi1yZWFjdC1yZWZyZXNoJ1xuaW1wb3J0IHRzZXNsaW50IGZyb20gJ3R5cGVzY3JpcHQtZXNsaW50J1xuXG5leHBvcnQgZGVmYXVsdCBbXG4gIHsgaWdub3JlczogWydkaXN0J10gfSxcbiAge1xuICAgIGZpbGVzOiBbJyoqLyoue3RzLHRzeH0nXSxcbiAgICBsYW5ndWFnZU9wdGlvbnM6IHtcbiAgICAgIGVjbWFWZXJzaW9uOiAyMDIwLFxuICAgICAgZ2xvYmFsczogZ2xvYmFscy5icm93c2VyLFxuICAgIH0sXG4gICAgcGx1Z2luczoge1xuICAgICAgJ3JlYWN0LWhvb2tzJzogcmVhY3RIb29rcyxcbiAgICAgICdyZWFjdC1yZWZyZXNoJzogcmVhY3RSZWZyZXNoLFxuICAgIH0sXG4gICAgcnVsZXM6IHtcbiAgICAgIC4uLnJlYWN0SG9va3MuY29uZmlncy5yZWNvbW1lbmRlZC5ydWxlcyxcbiAgICAgICdyZWFjdC1yZWZyZXNoL29ubHktZXhwb3J0LWNvbXBvbmVudHMnOiBbXG4gICAgICAgICd3YXJuJyxcbiAgICAgICAgeyBhbGxvd0NvbnN0YW50RXhwb3J0OiB0cnVlIH0sXG4gICAgICBdLFxuICAgIH0sXG4gIH0sXG4gIGpzLmNvbmZpZ3MucmVjb21tZW5kZWQsXG4gIC4uLnRzZXNsaW50LmNvbmZpZ3MucmVjb21tZW5kZWQsXG5dXG5gO1xuICB9XG59XG5cbi8qKlxuICogQ3JlYXRlcyB0aGUgdml0ZS5jb25maWcudHMgZmlsZVxuICovXG5jbGFzcyBWaXRlQ29uZmlnRmlsZSBleHRlbmRzIEZpbGVCYXNlIHtcbiAgY29uc3RydWN0b3IocHJvamVjdDogUHJvamVjdCkge1xuICAgIHN1cGVyKHByb2plY3QsICd2aXRlLmNvbmZpZy50cycsIHtcbiAgICAgIG1hcmtlcjogdHJ1ZSxcbiAgICAgIHJlYWRvbmx5OiB0cnVlLFxuICAgIH0pO1xuICB9XG5cbiAgc3ludGhlc2l6ZUNvbnRlbnQoKTogc3RyaW5nIHwgdW5kZWZpbmVkIHtcbiAgICByZXR1cm4gYGltcG9ydCB7IGRlZmluZUNvbmZpZyB9IGZyb20gJ3ZpdGUnXG5pbXBvcnQgcmVhY3QgZnJvbSAnQHZpdGVqcy9wbHVnaW4tcmVhY3QnXG5cbi8vIGh0dHBzOi8vdml0ZS5kZXYvY29uZmlnL1xuZXhwb3J0IGRlZmF1bHQgZGVmaW5lQ29uZmlnKHtcbiAgcGx1Z2luczogW3JlYWN0KCldLFxufSlcbmA7XG4gIH1cbn1cblxuLyoqXG4gKiBDcmVhdGVzIHRoZSBwb3N0Y3NzLmNvbmZpZy5qcyBmaWxlIGZvciBUYWlsd2luZCBDU1MgdjRcbiAqL1xuY2xhc3MgUG9zdENTU0NvbmZpZ0ZpbGUgZXh0ZW5kcyBGaWxlQmFzZSB7XG4gIGNvbnN0cnVjdG9yKHByb2plY3Q6IFByb2plY3QpIHtcbiAgICBzdXBlcihwcm9qZWN0LCAncG9zdGNzcy5jb25maWcuanMnLCB7XG4gICAgICBtYXJrZXI6IHRydWUsXG4gICAgICByZWFkb25seTogdHJ1ZSxcbiAgICB9KTtcbiAgfVxuXG4gIHN5bnRoZXNpemVDb250ZW50KCk6IHN0cmluZyB8IHVuZGVmaW5lZCB7XG4gICAgcmV0dXJuIGBleHBvcnQgZGVmYXVsdCB7XG4gIHBsdWdpbnM6IHtcbiAgICBcIkB0YWlsd2luZGNzcy9wb3N0Y3NzXCI6IHt9LFxuICAgIGF1dG9wcmVmaXhlcjoge30sXG4gIH0sXG59XG5gO1xuICB9XG59XG5cbmludGVyZmFjZSBWaXRlRW52RHRzT3B0aW9ucyB7XG4gIHJlYWRvbmx5IHNyY2Rpcjogc3RyaW5nO1xufVxuXG4vKipcbiAqIENyZWF0ZXMgdGhlIHZpdGUtZW52LmQudHMgZmlsZSBmb3IgdHlwZSBkZWNsYXJhdGlvbnNcbiAqL1xuY2xhc3MgVml0ZUVudkR0cyBleHRlbmRzIEZpbGVCYXNlIHtcbiAgY29uc3RydWN0b3IocHJvamVjdDogUHJvamVjdCwgb3B0aW9uczogVml0ZUVudkR0c09wdGlvbnMpIHtcbiAgICBzdXBlcihwcm9qZWN0LCBgJHtvcHRpb25zLnNyY2Rpcn0vdml0ZS1lbnYuZC50c2AsIHtcbiAgICAgIG1hcmtlcjogdHJ1ZSxcbiAgICAgIHJlYWRvbmx5OiB0cnVlLFxuICAgIH0pO1xuICB9XG5cbiAgc3ludGhlc2l6ZUNvbnRlbnQoKTogc3RyaW5nIHwgdW5kZWZpbmVkIHtcbiAgICByZXR1cm4gYC8vLyA8cmVmZXJlbmNlIHR5cGVzPVwidml0ZS9jbGllbnRcIiAvPlxuYDtcbiAgfVxufVxuXG5pbnRlcmZhY2UgQ29uZmlnRmlsZU9wdGlvbnMge1xuICByZWFkb25seSBzcmNkaXI6IHN0cmluZztcbn1cblxuLyoqXG4gKiBDcmVhdGVzIHRoZSBjb25maWcudHMgZmlsZSBmb3IgcnVudGltZSBjb25maWd1cmF0aW9uXG4gKi9cbmNsYXNzIENvbmZpZ0ZpbGUgZXh0ZW5kcyBGaWxlQmFzZSB7XG4gIGNvbnN0cnVjdG9yKHByb2plY3Q6IFByb2plY3QsIG9wdGlvbnM6IENvbmZpZ0ZpbGVPcHRpb25zKSB7XG4gICAgc3VwZXIocHJvamVjdCwgYCR7b3B0aW9ucy5zcmNkaXJ9L2NvbmZpZy50c2AsIHtcbiAgICAgIG1hcmtlcjogdHJ1ZSxcbiAgICAgIHJlYWRvbmx5OiB0cnVlLFxuICAgIH0pO1xuICB9XG5cbiAgc3ludGhlc2l6ZUNvbnRlbnQoKTogc3RyaW5nIHwgdW5kZWZpbmVkIHtcbiAgICAvLyBHZXQgY3VzdG9tIGNvbmZpZyBmaWVsZHMgZnJvbSBwYXJlbnQgcHJvamVjdFxuICAgIGNvbnN0IHZpdGVQcm9qZWN0ID0gdGhpcy5wcm9qZWN0IGFzIFZpdGVSZWFjdFByb2plY3Q7XG4gICAgY29uc3QgY3VzdG9tRmllbGRzID0gdml0ZVByb2plY3QuZ2V0Q29uZmlnRmllbGRzKCk7XG5cbiAgICAvLyBHZW5lcmF0ZSBiYXNlIGNvbmZpZyB0eXBlXG4gICAgY29uc3QgYmFzZUNvbmZpZ1R5cGUgPSBgdHlwZSBCYXNlQ29uZmlnID0ge1xuICBjbGVya1B1Ymxpc2hhYmxlS2V5OiBzdHJpbmc7XG4gIGFwaUJhc2VVcmw6IHN0cmluZztcbiAgdGltZW91dDogbnVtYmVyO1xufWA7XG5cbiAgICAvLyBHZW5lcmF0ZSBjdXN0b20gZmllbGRzIHR5cGUgaWYgYW55IGV4aXN0XG4gICAgbGV0IGNvbmZpZ1R5cGVEZWZpbml0aW9uOiBzdHJpbmc7XG4gICAgaWYgKGN1c3RvbUZpZWxkcy5zaXplID4gMCkge1xuICAgICAgY29uc3QgZmllbGREZWZpbml0aW9ucyA9IEFycmF5LmZyb20oY3VzdG9tRmllbGRzLmVudHJpZXMoKSlcbiAgICAgICAgLm1hcCgoW25hbWUsIHR5cGVdKSA9PiBgICAke25hbWV9OiAke3R5cGV9O2ApXG4gICAgICAgIC5qb2luKCdcXG4nKTtcblxuICAgICAgY29uc3QgY3VzdG9tRmllbGRzVHlwZSA9IGB0eXBlIEN1c3RvbUNvbmZpZyA9IHtcbiR7ZmllbGREZWZpbml0aW9uc31cbn1gO1xuXG4gICAgICBjb25maWdUeXBlRGVmaW5pdGlvbiA9IGAke2N1c3RvbUZpZWxkc1R5cGV9XG5cbmV4cG9ydCB0eXBlIENvbmZpZyA9IEJhc2VDb25maWcgJiBDdXN0b21Db25maWc7YDtcbiAgICB9IGVsc2Uge1xuICAgICAgY29uZmlnVHlwZURlZmluaXRpb24gPSAnZXhwb3J0IHR5cGUgQ29uZmlnID0gQmFzZUNvbmZpZzsnO1xuICAgIH1cblxuICAgIHJldHVybiBgaW1wb3J0IHsgdXNlRWZmZWN0LCB1c2VTdGF0ZSB9IGZyb20gJ3JlYWN0JztcblxuJHtiYXNlQ29uZmlnVHlwZX1cblxuJHtjb25maWdUeXBlRGVmaW5pdGlvbn1cblxuZXhwb3J0IGZ1bmN0aW9uIHVzZUNvbmZpZygpIHtcbiAgY29uc3QgW2xvYWRlZENvbmZpZywgc2V0TG9hZGVkQ29uZmlnXSA9IHVzZVN0YXRlPENvbmZpZ3x1bmRlZmluZWQ+KCk7XG4gIGNvbnN0IFtpc0xvYWRpbmcsIHNldElzTG9hZGluZ10gPSB1c2VTdGF0ZShmYWxzZSk7XG5cbiAgdXNlRWZmZWN0KCgpID0+IHtcbiAgICBpZiAoIWxvYWRlZENvbmZpZykge1xuICAgICAgZmV0Y2goJy9jb25maWcuanNvbicsIHsgY2FjaGU6ICduby1zdG9yZScgfSkudGhlbihyZXNwb25zZT0+e1xuICAgICAgICByZXNwb25zZS5qc29uKCkudGhlbigoY29uZmlnKSA9PiB7XG4gICAgICAgICAgc2V0SXNMb2FkaW5nKGZhbHNlKVxuICAgICAgICAgIHNldExvYWRlZENvbmZpZyhjb25maWcpXG4gICAgICAgIH0pO1xuICAgICAgfSk7XG5cbiAgICB9XG4gIH0sIFtsb2FkZWRDb25maWddKTtcblxuICByZXR1cm4geyBjb25maWc6IGxvYWRlZENvbmZpZywgaXNMb2FkaW5nIH07XG59XG5gO1xuICB9XG59XG5cbmludGVyZmFjZSBDb25maWdKc29uRmlsZU9wdGlvbnMge31cblxuLyoqXG4gKiBDcmVhdGVzIHRoZSBjb25maWcuanNvbiBzYW1wbGUgZmlsZSBmb3IgcnVudGltZSBjb25maWd1cmF0aW9uXG4gKi9cbmNsYXNzIENvbmZpZ0pzb25GaWxlIGV4dGVuZHMgU2FtcGxlRmlsZSB7XG4gIHBhdGg6IHN0cmluZztcbiAgY29uc3RydWN0b3IocHJvamVjdDogVml0ZVJlYWN0UHJvamVjdCwgX29wdGlvbnM6IENvbmZpZ0pzb25GaWxlT3B0aW9ucykge1xuICAgIHN1cGVyKHByb2plY3QsIGAke3Byb2plY3QucHVibGljRGlyfS9jb25maWcuanNvbmAsIHtcbiAgICAgIGNvbnRlbnRzOiBKU09OLnN0cmluZ2lmeSh7XG4gICAgICAgIGFwaUJhc2VVcmw6ICdodHRwOi8vbG9jYWxob3N0OjMwMDAvYXBpJyxcbiAgICAgICAgY2xlcmtQdWJsaXNoYWJsZUtleTogJ3RiZCcsXG4gICAgICB9LCBudWxsLCAyKSxcbiAgICB9KTtcbiAgICB0aGlzLnBhdGggPSBgJHtwcm9qZWN0LnB1YmxpY0Rpcn0vY29uZmlnLmpzb25gO1xuICB9XG59XG5cbmludGVyZmFjZSBJbmRleEh0bWxPcHRpb25zIHtcbiAgcmVhZG9ubHkgdGl0bGU6IHN0cmluZztcbiAgcmVhZG9ubHkgc3JjZGlyOiBzdHJpbmc7XG59XG5cbi8qKlxuICogQ3JlYXRlcyB0aGUgaW5kZXguaHRtbCBmaWxlXG4gKi9cbmNsYXNzIEluZGV4SHRtbCBleHRlbmRzIEZpbGVCYXNlIHtcbiAgcHJpdmF0ZSByZWFkb25seSB0aXRsZTogc3RyaW5nO1xuICBwcml2YXRlIHJlYWRvbmx5IHNyY2Rpcjogc3RyaW5nO1xuXG4gIGNvbnN0cnVjdG9yKHByb2plY3Q6IFByb2plY3QsIG9wdGlvbnM6IEluZGV4SHRtbE9wdGlvbnMpIHtcbiAgICBzdXBlcihwcm9qZWN0LCAnaW5kZXguaHRtbCcsIHtcbiAgICAgIG1hcmtlcjogdHJ1ZSxcbiAgICAgIHJlYWRvbmx5OiB0cnVlLFxuICAgIH0pO1xuXG4gICAgdGhpcy50aXRsZSA9IG9wdGlvbnMudGl0bGU7XG4gICAgdGhpcy5zcmNkaXIgPSBvcHRpb25zLnNyY2RpcjtcbiAgfVxuXG4gIHN5bnRoZXNpemVDb250ZW50KCk6IHN0cmluZyB8IHVuZGVmaW5lZCB7XG4gICAgcmV0dXJuIGA8IWRvY3R5cGUgaHRtbD5cbjxodG1sIGxhbmc9XCJlblwiPlxuICA8aGVhZD5cbiAgICA8bWV0YSBjaGFyc2V0PVwiVVRGLThcIiAvPlxuICAgIDxsaW5rIHJlbD1cImljb25cIiB0eXBlPVwiaW1hZ2Uvc3ZnK3htbFwiIGhyZWY9XCIvdml0ZS5zdmdcIiAvPlxuICAgIDxtZXRhIG5hbWU9XCJ2aWV3cG9ydFwiIGNvbnRlbnQ9XCJ3aWR0aD1kZXZpY2Utd2lkdGgsIGluaXRpYWwtc2NhbGU9MS4wXCIgLz5cbiAgICA8dGl0bGU+JHt0aGlzLnRpdGxlfTwvdGl0bGU+XG4gIDwvaGVhZD5cbiAgPGJvZHk+XG4gICAgPGRpdiBpZD1cInJvb3RcIj48L2Rpdj5cbiAgICA8c2NyaXB0IHR5cGU9XCJtb2R1bGVcIiBzcmM9XCIvJHt0aGlzLnNyY2Rpcn0vbWFpbi50c3hcIj48L3NjcmlwdD5cbiAgPC9ib2R5PlxuPC9odG1sPlxuYDtcbiAgfVxufVxuXG5pbnRlcmZhY2UgVml0ZVRzQ29uZmlnc09wdGlvbnMge1xuICByZWFkb25seSBzcmNkaXI6IHN0cmluZztcbn1cblxuLyoqXG4gKiBDcmVhdGVzIHRoZSBUeXBlU2NyaXB0IGNvbmZpZ3VyYXRpb24gZmlsZXMgZm9yIFZpdGVcbiAqL1xuY2xhc3MgVml0ZVRzQ29uZmlncyBleHRlbmRzIENvbXBvbmVudCB7XG4gIGNvbnN0cnVjdG9yKHByb2plY3Q6IFByb2plY3QsIG9wdGlvbnM6IFZpdGVUc0NvbmZpZ3NPcHRpb25zKSB7XG4gICAgc3VwZXIocHJvamVjdCk7XG5cbiAgICAvLyBNYWluIHRzY29uZmlnLmpzb24gKHByb2plY3QgcmVmZXJlbmNlcylcbiAgICBuZXcgSnNvbkZpbGUocHJvamVjdCwgJ3RzY29uZmlnLmpzb24nLCB7XG4gICAgICBtYXJrZXI6IHRydWUsXG4gICAgICByZWFkb25seTogdHJ1ZSxcbiAgICAgIG9iajoge1xuICAgICAgICBmaWxlczogW10sXG4gICAgICAgIHJlZmVyZW5jZXM6IFtcbiAgICAgICAgICB7IHBhdGg6ICcuL3RzY29uZmlnLmFwcC5qc29uJyB9LFxuICAgICAgICAgIHsgcGF0aDogJy4vdHNjb25maWcubm9kZS5qc29uJyB9LFxuICAgICAgICBdLFxuICAgICAgfSxcbiAgICB9KTtcblxuICAgIC8vIHRzY29uZmlnLmFwcC5qc29uIChmb3IgYXBwbGljYXRpb24gY29kZSlcbiAgICBuZXcgSnNvbkZpbGUocHJvamVjdCwgJ3RzY29uZmlnLmFwcC5qc29uJywge1xuICAgICAgbWFya2VyOiB0cnVlLFxuICAgICAgcmVhZG9ubHk6IHRydWUsXG4gICAgICBvYmo6IHtcbiAgICAgICAgY29tcGlsZXJPcHRpb25zOiB7XG4gICAgICAgICAgdGFyZ2V0OiAnRVMyMDIwJyxcbiAgICAgICAgICB1c2VEZWZpbmVGb3JDbGFzc0ZpZWxkczogdHJ1ZSxcbiAgICAgICAgICBsaWI6IFsnRVMyMDIwJywgJ0RPTScsICdET00uSXRlcmFibGUnXSxcbiAgICAgICAgICBtb2R1bGU6ICdFU05leHQnLFxuICAgICAgICAgIHNraXBMaWJDaGVjazogdHJ1ZSxcblxuICAgICAgICAgIC8vIEJ1bmRsZXIgbW9kZVxuICAgICAgICAgIG1vZHVsZVJlc29sdXRpb246ICdidW5kbGVyJyxcbiAgICAgICAgICBhbGxvd0ltcG9ydGluZ1RzRXh0ZW5zaW9uczogdHJ1ZSxcbiAgICAgICAgICBpc29sYXRlZE1vZHVsZXM6IHRydWUsXG4gICAgICAgICAgbW9kdWxlRGV0ZWN0aW9uOiAnZm9yY2UnLFxuICAgICAgICAgIG5vRW1pdDogdHJ1ZSxcbiAgICAgICAgICBqc3g6ICdyZWFjdC1qc3gnLFxuXG4gICAgICAgICAgLy8gTGludGluZ1xuICAgICAgICAgIHN0cmljdDogdHJ1ZSxcbiAgICAgICAgICBub1VudXNlZExvY2FsczogdHJ1ZSxcbiAgICAgICAgICBub1VudXNlZFBhcmFtZXRlcnM6IHRydWUsXG4gICAgICAgICAgbm9GYWxsdGhyb3VnaENhc2VzSW5Td2l0Y2g6IHRydWUsXG4gICAgICAgICAgbm9VbmNoZWNrZWRTaWRlRWZmZWN0SW1wb3J0czogdHJ1ZSxcbiAgICAgICAgfSxcbiAgICAgICAgaW5jbHVkZTogW29wdGlvbnMuc3JjZGlyXSxcbiAgICAgIH0sXG4gICAgfSk7XG5cbiAgICAvLyB0c2NvbmZpZy5ub2RlLmpzb24gKGZvciBWaXRlIGNvbmZpZylcbiAgICBuZXcgSnNvbkZpbGUocHJvamVjdCwgJ3RzY29uZmlnLm5vZGUuanNvbicsIHtcbiAgICAgIG1hcmtlcjogdHJ1ZSxcbiAgICAgIHJlYWRvbmx5OiB0cnVlLFxuICAgICAgb2JqOiB7XG4gICAgICAgIGNvbXBpbGVyT3B0aW9uczoge1xuICAgICAgICAgIHRhcmdldDogJ0VTMjAyMicsXG4gICAgICAgICAgbGliOiBbJ0VTMjAyMyddLFxuICAgICAgICAgIG1vZHVsZTogJ0VTTmV4dCcsXG4gICAgICAgICAgc2tpcExpYkNoZWNrOiB0cnVlLFxuXG4gICAgICAgICAgLy8gQnVuZGxlciBtb2RlXG4gICAgICAgICAgbW9kdWxlUmVzb2x1dGlvbjogJ2J1bmRsZXInLFxuICAgICAgICAgIGFsbG93SW1wb3J0aW5nVHNFeHRlbnNpb25zOiB0cnVlLFxuICAgICAgICAgIGlzb2xhdGVkTW9kdWxlczogdHJ1ZSxcbiAgICAgICAgICBtb2R1bGVEZXRlY3Rpb246ICdmb3JjZScsXG4gICAgICAgICAgbm9FbWl0OiB0cnVlLFxuXG4gICAgICAgICAgLy8gTGludGluZ1xuICAgICAgICAgIHN0cmljdDogdHJ1ZSxcbiAgICAgICAgICBub1VudXNlZExvY2FsczogdHJ1ZSxcbiAgICAgICAgICBub1VudXNlZFBhcmFtZXRlcnM6IHRydWUsXG4gICAgICAgICAgbm9GYWxsdGhyb3VnaENhc2VzSW5Td2l0Y2g6IHRydWUsXG4gICAgICAgIH0sXG4gICAgICAgIGluY2x1ZGU6IFsndml0ZS5jb25maWcudHMnXSxcbiAgICAgIH0sXG4gICAgfSk7XG4gIH1cbn1cblxuaW50ZXJmYWNlIFZpdGVTYW1wbGVDb2RlT3B0aW9ucyB7XG4gIHJlYWRvbmx5IHNyY2Rpcjogc3RyaW5nO1xuICByZWFkb25seSBwdWJsaWNEaXI6IHN0cmluZztcbn1cblxuLyoqXG4gKiBHZW5lcmF0ZXMgc2FtcGxlIGNvZGUgZm9yIGEgVml0ZSArIFJlYWN0IHByb2plY3RcbiAqL1xuY2xhc3MgVml0ZVNhbXBsZUNvZGUgZXh0ZW5kcyBDb21wb25lbnQge1xuICBjb25zdHJ1Y3Rvcihwcm9qZWN0OiBQcm9qZWN0LCBvcHRpb25zOiBWaXRlU2FtcGxlQ29kZU9wdGlvbnMpIHtcbiAgICBzdXBlcihwcm9qZWN0KTtcblxuICAgIC8vIENyZWF0ZSBBcHAudHN4XG4gICAgbmV3IFNhbXBsZUZpbGUocHJvamVjdCwgYCR7b3B0aW9ucy5zcmNkaXJ9L0FwcC50c3hgLCB7XG4gICAgICBjb250ZW50czogYGltcG9ydCAnLi9BcHAuY3NzJ1xuaW1wb3J0IHsgU2lnbmVkSW4sIFVzZXJCdXR0b24gfSBmcm9tICdAY2xlcmsvY2xlcmstcmVhY3QnO1xuaW1wb3J0IHsgdXNlQmFzZVF1ZXJ5IH0gZnJvbSBcIi4vc2VydmljZXMvQmFzZVNlcnZpY2UudHNcIjtcblxuXG5mdW5jdGlvbiBBcHAoKSB7XG4gIGNvbnN0IHtkYXRhLCBpc0xvYWRpbmd9ID0gdXNlQmFzZVF1ZXJ5KCcvJywge3JldHJ5T25Nb3VudDogdHJ1ZX0pO1xuXG5cbiAgcmV0dXJuIChcbiAgICA8PlxuICAgICAgICA8aGVhZGVyPlxuXG4gICAgICAgICAgPFNpZ25lZEluPlxuICAgICAgICAgICAgPFVzZXJCdXR0b24vPlxuICAgICAgICAgIDwvU2lnbmVkSW4+XG4gICAgICAgIDwvaGVhZGVyPlxuICAgICAgICA8ZGl2IGNsYXNzTmFtZT1cImNhcmRcIj5cbiAgICAgICAgICB7IGlzTG9hZGluZyA/ICdMb2FkaW5nLi4uJyA6IEpTT04uc3RyaW5naWZ5KGRhdGEsIG51bGwgLCl9XG4gICAgICAgIDwvZGl2PlxuICAgIDwvPlxuICApXG59XG5cbmV4cG9ydCBkZWZhdWx0IEFwcFxuYCxcbiAgICB9KTtcblxuICAgIC8vIENyZWF0ZSBCYXNlU2VydmljZS50c1xuICAgIG5ldyBUZXh0RmlsZShwcm9qZWN0LCBgJHtvcHRpb25zLnNyY2Rpcn0vc2VydmljZXMvQmFzZVNlcnZpY2UudHNgLCB7XG4gICAgICBtYXJrZXI6IHRydWUsXG4gICAgICByZWFkb25seTogdHJ1ZSxcbiAgICAgIGxpbmVzOiBbXG4gICAgICAgICdpbXBvcnQgeyB1c2VRdWVyeSwgVXNlUXVlcnlPcHRpb25zIH0gZnJvbSBcXCdAdGFuc3RhY2svcmVhY3QtcXVlcnlcXCc7JyxcbiAgICAgICAgJ2ltcG9ydCB7IHVzZUF1dGggfSBmcm9tIFxcJ0BjbGVyay9jbGVyay1yZWFjdFxcJzsnLFxuICAgICAgICAnaW1wb3J0IHsgdXNlQ29uZmlnIH0gZnJvbSBcXCcuLi9jb25maWdcXCc7JyxcbiAgICAgICAgJycsXG4gICAgICAgICdleHBvcnQgZnVuY3Rpb24gdXNlQmFzZVF1ZXJ5PFREYXRhID0gdW5rbm93bj4oJyxcbiAgICAgICAgJyAgcGF0aDogc3RyaW5nLCcsXG4gICAgICAgICcgIG9wdGlvbnM/OiBPbWl0PFVzZVF1ZXJ5T3B0aW9uczxURGF0YT4sIFxcJ3F1ZXJ5S2V5XFwnIHwgXFwncXVlcnlGblxcJz4nLFxuICAgICAgICAnKSB7JyxcbiAgICAgICAgJyAgY29uc3QgeyBjb25maWcgfSA9IHVzZUNvbmZpZygpOycsXG4gICAgICAgICcgIGNvbnN0IHsgZ2V0VG9rZW4gfSA9IHVzZUF1dGgoKTsnLFxuICAgICAgICAnJyxcbiAgICAgICAgJyAgcmV0dXJuIHVzZVF1ZXJ5PFREYXRhPih7JyxcbiAgICAgICAgJyAgICBxdWVyeUtleTogW1xcJ2Jhc2VcXCddLCcsXG4gICAgICAgICcgICAgcXVlcnlGbjogYXN5bmMgKCkgPT4geycsXG4gICAgICAgICcgICAgICBpZiAoIWNvbmZpZz8uYXBpQmFzZVVybCkgeycsXG4gICAgICAgICcgICAgICAgIHRocm93IG5ldyBFcnJvcihcXCdBUEkgYmFzZSBVUkwgbm90IGNvbmZpZ3VyZWRcXCcpOycsXG4gICAgICAgICcgICAgICB9JyxcbiAgICAgICAgJycsXG4gICAgICAgICcgICAgICBjb25zdCB0b2tlbiA9IGF3YWl0IGdldFRva2VuKCk7JyxcbiAgICAgICAgJycsXG4gICAgICAgICcgICAgICAvLyBOb3JtYWxpemUgVVJMIGJ5IHJlbW92aW5nIHRyYWlsaW5nIHNsYXNoIGZyb20gYmFzZVVybCBhbmQgZW5zdXJpbmcgcGF0aCBzdGFydHMgd2l0aCBzbGFzaCcsXG4gICAgICAgICcgICAgICBjb25zdCBiYXNlVXJsID0gY29uZmlnLmFwaUJhc2VVcmwucmVwbGFjZSgvXFxcXC8rJC8sIFxcJ1xcJyk7JyxcbiAgICAgICAgJyAgICAgIGNvbnN0IG5vcm1hbGl6ZWRQYXRoID0gcGF0aC5zdGFydHNXaXRoKFxcJy9cXCcpID8gcGF0aCA6IGAvJHtwYXRofWA7JyxcbiAgICAgICAgJyAgICAgICcsXG4gICAgICAgICcgICAgICBjb25zdCByZXNwb25zZSA9IGF3YWl0IGZldGNoKGAke2Jhc2VVcmx9JHtub3JtYWxpemVkUGF0aH1gLCB7JyxcbiAgICAgICAgJyAgICAgICAgaGVhZGVyczogeycsXG4gICAgICAgICcgICAgICAgICAgXFwnQXV0aG9yaXphdGlvblxcJzogYEJlYXJlciAke3Rva2VufWAsJyxcbiAgICAgICAgJyAgICAgICAgfSwnLFxuICAgICAgICAnICAgICAgfSk7JyxcbiAgICAgICAgJycsXG4gICAgICAgICcgICAgICBpZiAoIXJlc3BvbnNlLm9rKSB7JyxcbiAgICAgICAgJyAgICAgICAgdGhyb3cgbmV3IEVycm9yKGBIVFRQIGVycm9yISBzdGF0dXM6ICR7cmVzcG9uc2Uuc3RhdHVzfWApOycsXG4gICAgICAgICcgICAgICB9JyxcbiAgICAgICAgJycsXG4gICAgICAgICcgICAgICByZXR1cm4gcmVzcG9uc2UuanNvbigpOycsXG4gICAgICAgICcgICAgfSwnLFxuICAgICAgICAnICAgIGVuYWJsZWQ6ICEhY29uZmlnPy5hcGlCYXNlVXJsLCcsXG4gICAgICAgICcgICAgLi4ub3B0aW9ucywnLFxuICAgICAgICAnICB9KTsnLFxuICAgICAgICAnfScsXG4gICAgICBdLFxuICAgIH0pO1xuXG4gICAgLy8gQ3JlYXRlIFJvb3RTZXJ2aWNlLnRzXG4gICAgbmV3IFNhbXBsZUZpbGUocHJvamVjdCwgYCR7b3B0aW9ucy5zcmNkaXJ9L3NlcnZpY2VzL1Jvb3RTZXJ2aWNlLnRzYCwge1xuICAgICAgY29udGVudHM6IGBpbXBvcnQgeyBVc2VRdWVyeU9wdGlvbnMgfSBmcm9tICdAdGFuc3RhY2svcmVhY3QtcXVlcnknO1xuaW1wb3J0IHsgdXNlQmFzZVF1ZXJ5IH0gZnJvbSAnLi9CYXNlU2VydmljZSc7XG5cbmV4cG9ydCBmdW5jdGlvbiB1c2VSb290UXVlcnk8VERhdGEgPSB1bmtub3duPihcbiAgb3B0aW9ucz86IE9taXQ8VXNlUXVlcnlPcHRpb25zPFREYXRhPiwgJ3F1ZXJ5S2V5JyB8ICdxdWVyeUZuJz5cbikge1xuICByZXR1cm4gdXNlQmFzZVF1ZXJ5PFREYXRhPignLycsIG9wdGlvbnMpO1xufVxuYCxcbiAgICB9KTtcblxuICAgIC8vIENyZWF0ZSBBcHAuY3NzXG4gICAgbmV3IFNhbXBsZUZpbGUocHJvamVjdCwgYCR7b3B0aW9ucy5zcmNkaXJ9L0FwcC5jc3NgLCB7XG4gICAgICBjb250ZW50czogYCNyb290IHtcbiAgd2lkdGg6IDEwMCU7XG4gIGhlaWdodDogMTAwJTtcbn1cblxuLmxvZ28ge1xuICBoZWlnaHQ6IDZlbTtcbiAgcGFkZGluZzogMS41ZW07XG4gIHdpbGwtY2hhbmdlOiBmaWx0ZXI7XG4gIHRyYW5zaXRpb246IGZpbHRlciAzMDBtcztcbn1cbi5sb2dvOmhvdmVyIHtcbiAgZmlsdGVyOiBkcm9wLXNoYWRvdygwIDAgMmVtICM2NDZjZmZhYSk7XG59XG4ubG9nby5yZWFjdDpob3ZlciB7XG4gIGZpbHRlcjogZHJvcC1zaGFkb3coMCAwIDJlbSAjNjFkYWZiYWEpO1xufVxuXG5Aa2V5ZnJhbWVzIGxvZ28tc3BpbiB7XG4gIGZyb20ge1xuICAgIHRyYW5zZm9ybTogcm90YXRlKDBkZWcpO1xuICB9XG4gIHRvIHtcbiAgICB0cmFuc2Zvcm06IHJvdGF0ZSgzNjBkZWcpO1xuICB9XG59XG5cbkBtZWRpYSAocHJlZmVycy1yZWR1Y2VkLW1vdGlvbjogbm8tcHJlZmVyZW5jZSkge1xuICBhOm50aC1vZi10eXBlKDIpIC5sb2dvIHtcbiAgICBhbmltYXRpb246IGxvZ28tc3BpbiBpbmZpbml0ZSAyMHMgbGluZWFyO1xuICB9XG59XG5cbi5jYXJkIHtcbiAgcGFkZGluZzogMmVtO1xufVxuXG4ucmVhZC10aGUtZG9jcyB7XG4gIGNvbG9yOiAjODg4O1xufVxuYCxcbiAgICB9KTtcblxuICAgIC8vIENyZWF0ZSBtYWluLnRzeFxuICAgIG5ldyBTYW1wbGVGaWxlKHByb2plY3QsIGAke29wdGlvbnMuc3JjZGlyfS9tYWluLnRzeGAsIHtcbiAgICAgIGNvbnRlbnRzOiBgaW1wb3J0IHsgU3RyaWN0TW9kZSB9IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHsgY3JlYXRlUm9vdCB9IGZyb20gJ3JlYWN0LWRvbS9jbGllbnQnXG5pbXBvcnQgJy4vaW5kZXguY3NzJ1xuaW1wb3J0IEFwcCBmcm9tICcuL0FwcC50c3gnXG5pbXBvcnQgeyBDbGVya1Byb3ZpZGVyLCBTaWduZWRJbiwgU2lnbmVkT3V0LCBTaWduSW5CdXR0b24gfSBmcm9tICdAY2xlcmsvY2xlcmstcmVhY3QnXG5pbXBvcnQgeyB1c2VDb25maWcgfSBmcm9tICcuL2NvbmZpZydcbmltcG9ydCB7IFF1ZXJ5Q2xpZW50LCBRdWVyeUNsaWVudFByb3ZpZGVyIH0gZnJvbSBcIkB0YW5zdGFjay9yZWFjdC1xdWVyeVwiXG5cbmZ1bmN0aW9uIEFwcFdyYXBwZXIoKSB7XG4gIGNvbnN0IHsgY29uZmlnLCBpc0xvYWRpbmcgfSA9IHVzZUNvbmZpZygpO1xuICBjb25zdCBxdWVyeUNsaWVudCA9IG5ldyBRdWVyeUNsaWVudCgpXG4gIGlmIChpc0xvYWRpbmcgfHwgIWNvbmZpZykge1xuICAgIHJldHVybiAoXG4gICAgICA8ZGl2IGNsYXNzTmFtZT1cImxvYWRpbmctc2NyZWVuXCI+XG4gICAgICAgIExvYWRpbmcuLi5cbiAgICAgIDwvZGl2PlxuICAgICk7XG4gIH1cblxuICByZXR1cm4gKFxuICAgIDxDbGVya1Byb3ZpZGVyIHB1Ymxpc2hhYmxlS2V5PXtjb25maWcuY2xlcmtQdWJsaXNoYWJsZUtleX0+XG4gICAgICA8UXVlcnlDbGllbnRQcm92aWRlciBjbGllbnQ9e3F1ZXJ5Q2xpZW50fT5cbiAgICAgICAgPGRpdiBjbGFzc05hbWU9XCJ3LWZ1bGwgaC1mdWxsXCI+XG4gICAgICAgICAgPFNpZ25lZE91dD5cbiAgICAgICAgICAgIDxkaXYgY2xhc3NOYW1lPVwiZmxleCBpdGVtcy1jZW50ZXIganVzdGlmeS1jZW50ZXIgbWluLWgtc2NyZWVuXCI+XG4gICAgICAgICAgICAgIDxTaWduSW5CdXR0b24vPlxuICAgICAgICAgICAgPC9kaXY+XG4gICAgICAgICAgPC9TaWduZWRPdXQ+XG4gICAgICAgICAgPFNpZ25lZEluPlxuICAgICAgICAgICAgPEFwcC8+XG4gICAgICAgICAgPC9TaWduZWRJbj5cbiAgICAgICAgPC9kaXY+XG4gICAgICA8L1F1ZXJ5Q2xpZW50UHJvdmlkZXI+XG4gICAgPC9DbGVya1Byb3ZpZGVyPlxuICApO1xufVxuXG5jcmVhdGVSb290KGRvY3VtZW50LmdldEVsZW1lbnRCeUlkKCdyb290JykhKS5yZW5kZXIoXG4gIDxTdHJpY3RNb2RlPlxuICAgIDxBcHBXcmFwcGVyLz5cbiAgPC9TdHJpY3RNb2RlPlxuKVxuYCxcbiAgICB9KTtcblxuICAgIC8vIENyZWF0ZSBpbmRleC5jc3NcbiAgICBuZXcgU2FtcGxlRmlsZShwcm9qZWN0LCBgJHtvcHRpb25zLnNyY2Rpcn0vaW5kZXguY3NzYCwge1xuICAgICAgY29udGVudHM6IGBAaW1wb3J0IFwidGFpbHdpbmRjc3NcIjtcblxuQHRoZW1lIHtcbiAgLS1jb2xvci1zY2hlbWU6IGxpZ2h0IGRhcms7XG59XG5cbjpyb290IHtcbiAgZm9udC1mYW1pbHk6IEludGVyLCBzeXN0ZW0tdWksIEF2ZW5pciwgSGVsdmV0aWNhLCBBcmlhbCwgc2Fucy1zZXJpZjtcbiAgbGluZS1oZWlnaHQ6IDEuNTtcbiAgZm9udC13ZWlnaHQ6IDQwMDtcblxuICBjb2xvci1zY2hlbWU6IGxpZ2h0IGRhcms7XG5cbiAgZm9udC1zeW50aGVzaXM6IG5vbmU7XG4gIHRleHQtcmVuZGVyaW5nOiBvcHRpbWl6ZUxlZ2liaWxpdHk7XG4gIC13ZWJraXQtZm9udC1zbW9vdGhpbmc6IGFudGlhbGlhc2VkO1xuICAtbW96LW9zeC1mb250LXNtb290aGluZzogZ3JheXNjYWxlO1xufVxuXG5hIHtcbiAgZm9udC13ZWlnaHQ6IDUwMDtcbiAgY29sb3I6ICM2NDZjZmY7XG4gIHRleHQtZGVjb3JhdGlvbjogaW5oZXJpdDtcbn1cbmE6aG92ZXIge1xuICBjb2xvcjogIzUzNWJmMjtcbn1cblxuYm9keSB7XG4gIG1hcmdpbjogMDtcbiAgbWluLXdpZHRoOiAzMjBweDtcbiAgbWluLWhlaWdodDogMTAwdmg7XG59XG5cbmgxIHtcbiAgZm9udC1zaXplOiAzLjJlbTtcbiAgbGluZS1oZWlnaHQ6IDEuMTtcbn1cblxuYnV0dG9uIHtcbiAgZm9udC1mYW1pbHk6IGluaGVyaXQ7XG4gIGN1cnNvcjogcG9pbnRlcjtcbn1cblxuLmxvYWRpbmctc2NyZWVuIHtcbiAgZGlzcGxheTogZmxleDtcbiAganVzdGlmeS1jb250ZW50OiBjZW50ZXI7XG4gIGFsaWduLWl0ZW1zOiBjZW50ZXI7XG4gIGhlaWdodDogMTAwdmg7XG4gIGZvbnQtc2l6ZTogMS41cmVtO1xufVxuXG5odG1sLCBib2R5LCAjcm9vdCB7XG4gIHdpZHRoOiAxMDAlO1xuICBoZWlnaHQ6IDEwMCU7XG59XG5gLFxuICAgIH0pO1xuXG4gIH1cbn1cbiJdfQ==
|