siarashield_workspace 0.0.33 → 0.0.36
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/README.md +105 -195
- package/esm2022/lib/siara-shield-script-utils.mjs +3 -2
- package/esm2022/lib/siara-shield-vendor-runtime.mjs +98 -0
- package/esm2022/lib/siara-shield.component.mjs +29 -15
- package/esm2022/lib/siara-shield.mjs +29 -15
- package/fesm2022/siarashield_workspace.mjs +182 -57
- package/fesm2022/siarashield_workspace.mjs.map +1 -1
- package/lib/siara-shield-vendor-runtime.d.ts +1 -0
- package/lib/siara-shield.component.d.ts +3 -1
- package/lib/siara-shield.d.ts +3 -1
- package/package.json +1 -1
- package/siarashield_workspace-0.0.36.tgz +0 -0
- package/siarashield_workspace-0.0.33.tgz +0 -0
package/README.md
CHANGED
|
@@ -1,49 +1,31 @@
|
|
|
1
|
-
|
|
1
|
+
# `siarashield_workspace`
|
|
2
2
|
|
|
3
3
|
Angular integration for CyberSiara **SiaraShield** captcha.
|
|
4
4
|
|
|
5
5
|
## Angular compatibility
|
|
6
6
|
|
|
7
|
-
- Minimum supported Angular
|
|
8
|
-
- Supported
|
|
9
|
-
- Peer
|
|
7
|
+
- Minimum supported Angular: `16`
|
|
8
|
+
- Supported range: `16.x` to `21.x`
|
|
9
|
+
- Peer dependencies:
|
|
10
10
|
- `@angular/core >=16.0.0 <22.0.0`
|
|
11
11
|
- `@angular/common >=16.0.0 <22.0.0`
|
|
12
12
|
|
|
13
|
-
##
|
|
13
|
+
## Quick start (recommended, easiest path)
|
|
14
|
+
|
|
15
|
+
Use this flow first. Most users need only these steps.
|
|
16
|
+
|
|
17
|
+
1. Install package:
|
|
14
18
|
|
|
15
19
|
```bash
|
|
16
20
|
npm i siarashield_workspace
|
|
17
21
|
```
|
|
18
22
|
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
- **Recommended:** Angular component (`<siara-shield ...>`).
|
|
24
|
-
- **Alternative:** function API (`initSiaraShield(...)` + `checkSiaraShieldCaptcha...`).
|
|
25
|
-
|
|
26
|
-
Do not mix both styles on the same page.
|
|
27
|
-
|
|
28
|
-
1. Install package: `npm i siarashield_workspace`.
|
|
29
|
-
2. Put only the **public key** in Angular environment.
|
|
30
|
-
3. Keep the **private key** on backend only (`.env` or secret store).
|
|
31
|
-
4. Render captcha container (component tag or `<div class="SiaraShield"></div>` for function API).
|
|
32
|
-
5. Keep submit button class: `class="CaptchaSubmit"`.
|
|
33
|
-
6. Initialize captcha once per page load.
|
|
34
|
-
7. On submit, run captcha check before API call.
|
|
35
|
-
8. Call API only when captcha returns success (`result.ok === true` or `ok === true`).
|
|
36
|
-
|
|
37
|
-
## Get your keys
|
|
38
|
-
|
|
39
|
-
Get your public and private keys from <a href="https://mycybersiara.com" target="_blank" rel="noopener noreferrer">mycybersiara.com</a>.
|
|
40
|
-
|
|
41
|
-
## Put keys in the right place
|
|
23
|
+
2. Ensure Angular environment files exist.
|
|
24
|
+
- If your project does not have `src/environments`, create:
|
|
25
|
+
- `src/environments/environment.ts`
|
|
26
|
+
- `src/environments/environment.prod.ts`
|
|
42
27
|
|
|
43
|
-
|
|
44
|
-
- **Backend (.env):** private key only
|
|
45
|
-
|
|
46
|
-
Angular `environment` example:
|
|
28
|
+
3. Put only your **public key** in Angular environment:
|
|
47
29
|
|
|
48
30
|
```ts
|
|
49
31
|
export const environment = {
|
|
@@ -51,17 +33,7 @@ export const environment = {
|
|
|
51
33
|
};
|
|
52
34
|
```
|
|
53
35
|
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
```dotenv
|
|
57
|
-
SIARASHIELD_PRIVATE_KEY=YOUR-PRIVATE-KEY
|
|
58
|
-
```
|
|
59
|
-
|
|
60
|
-
Never place the private key in Angular environment files, templates, or browser code.
|
|
61
|
-
|
|
62
|
-
## Recommended: use the Angular component (easiest)
|
|
63
|
-
|
|
64
|
-
### 1) Add the component to your template
|
|
36
|
+
4. Render component + submit button in template:
|
|
65
37
|
|
|
66
38
|
```html
|
|
67
39
|
<siara-shield
|
|
@@ -73,7 +45,7 @@ Never place the private key in Angular environment files, templates, or browser
|
|
|
73
45
|
<button type="submit" class="CaptchaSubmit" (click)="onSubmit()">Submit</button>
|
|
74
46
|
```
|
|
75
47
|
|
|
76
|
-
|
|
48
|
+
5. Check captcha before API submit:
|
|
77
49
|
|
|
78
50
|
```ts
|
|
79
51
|
import { Component, ViewChild } from '@angular/core';
|
|
@@ -88,63 +60,76 @@ import { SiaraShieldComponent } from 'siarashield_workspace';
|
|
|
88
60
|
})
|
|
89
61
|
export class FormComponent {
|
|
90
62
|
protected readonly environment = environment;
|
|
91
|
-
|
|
92
|
-
// Recommended: set from server (see CSP section)
|
|
93
63
|
protected readonly cspNonce = (window as any).__cspNonce as string | undefined;
|
|
94
|
-
private isSubmitting = false;
|
|
95
64
|
|
|
65
|
+
private isSubmitting = false;
|
|
96
66
|
@ViewChild(SiaraShieldComponent) private readonly captcha?: SiaraShieldComponent;
|
|
97
67
|
|
|
98
68
|
onCaptchaToken(token: string) {
|
|
99
|
-
// Optional: use token immediately if needed
|
|
100
69
|
console.log('SiaraShield token:', token);
|
|
101
70
|
}
|
|
102
71
|
|
|
103
72
|
async onSubmit() {
|
|
104
73
|
if (this.isSubmitting) return;
|
|
105
74
|
this.isSubmitting = true;
|
|
106
|
-
|
|
107
75
|
try {
|
|
108
76
|
const ok = await this.captcha?.checkCaptchaAsync();
|
|
109
|
-
if (!ok)
|
|
110
|
-
console.log('Captcha not completed yet');
|
|
111
|
-
return;
|
|
112
|
-
}
|
|
77
|
+
if (!ok) return;
|
|
113
78
|
|
|
114
|
-
//
|
|
79
|
+
// Call your backend API only after captcha success.
|
|
115
80
|
alert('Form submitted successfully');
|
|
116
81
|
} finally {
|
|
117
|
-
// Allow next user attempt if captcha or API flow was not completed.
|
|
118
82
|
this.isSubmitting = false;
|
|
119
83
|
}
|
|
120
84
|
}
|
|
121
85
|
}
|
|
122
86
|
```
|
|
123
87
|
|
|
124
|
-
|
|
88
|
+
6. Keep your submit API logic only in the success branch (`ok === true`).
|
|
89
|
+
|
|
90
|
+
## Key handling (required)
|
|
91
|
+
|
|
92
|
+
- **Frontend (Angular): public key only**
|
|
93
|
+
- **Backend (.env or secret store): private key only**
|
|
125
94
|
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
-
|
|
95
|
+
Backend example:
|
|
96
|
+
|
|
97
|
+
```dotenv
|
|
98
|
+
SIARASHIELD_PRIVATE_KEY=YOUR-PRIVATE-KEY
|
|
99
|
+
```
|
|
130
100
|
|
|
131
|
-
|
|
101
|
+
Never place the private key in Angular code or browser-accessible files.
|
|
132
102
|
|
|
133
|
-
|
|
103
|
+
Get keys from [mycybersiara.com](https://mycybersiara.com).
|
|
104
|
+
|
|
105
|
+
## Integration rules (important)
|
|
106
|
+
|
|
107
|
+
- Choose one style per page:
|
|
108
|
+
- **Recommended:** `<siara-shield ...>`
|
|
109
|
+
- **Alternative:** function API
|
|
110
|
+
- Do not mix both styles on the same page.
|
|
111
|
+
- Keep submit button class: `CaptchaSubmit`.
|
|
112
|
+
- Initialize captcha once per page load.
|
|
113
|
+
|
|
114
|
+
## Function API (alternative)
|
|
115
|
+
|
|
116
|
+
Use this only when component integration is not possible.
|
|
117
|
+
|
|
118
|
+
Template:
|
|
134
119
|
|
|
135
120
|
```html
|
|
136
121
|
<div class="SiaraShield"></div>
|
|
137
122
|
<button type="submit" class="CaptchaSubmit" (click)="onSubmit()">Submit</button>
|
|
138
123
|
```
|
|
139
124
|
|
|
125
|
+
TypeScript:
|
|
126
|
+
|
|
140
127
|
```ts
|
|
141
128
|
import { environment } from '../environments/environment';
|
|
142
129
|
import { initSiaraShield, checkSiaraShieldCaptcha } from 'siarashield_workspace';
|
|
143
130
|
|
|
144
131
|
export class FormComponent {
|
|
145
132
|
ngOnInit() {
|
|
146
|
-
// Angular does not wait for async lifecycle hooks.
|
|
147
|
-
// Use void to run async init without blocking render.
|
|
148
133
|
void this.initializeCaptcha();
|
|
149
134
|
}
|
|
150
135
|
|
|
@@ -152,68 +137,47 @@ export class FormComponent {
|
|
|
152
137
|
await initSiaraShield({
|
|
153
138
|
publicKey: environment.siaraShieldPublicKey,
|
|
154
139
|
cspNonce: (window as any).__cspNonce || undefined,
|
|
155
|
-
// Optional: set true only while debugging vendor/runtime internals.
|
|
156
|
-
allowVendorConsoleLogs: false,
|
|
157
140
|
});
|
|
158
141
|
}
|
|
159
142
|
|
|
160
143
|
onSubmit() {
|
|
161
144
|
const result = checkSiaraShieldCaptcha();
|
|
162
|
-
if (!result.ok)
|
|
163
|
-
console.log('Captcha not completed yet');
|
|
164
|
-
return;
|
|
165
|
-
}
|
|
145
|
+
if (!result.ok) return;
|
|
166
146
|
|
|
167
147
|
console.log(result.token);
|
|
168
148
|
// API call here
|
|
169
|
-
alert('Form submitted successfully');
|
|
170
149
|
}
|
|
171
150
|
}
|
|
172
151
|
```
|
|
173
152
|
|
|
174
|
-
|
|
175
|
-
|
|
176
|
-
If your frontend reads nonce from the DOM instead of `window`, reuse nonce from an existing script tag:
|
|
177
|
-
|
|
178
|
-
```ts
|
|
179
|
-
const nonce = document.querySelector('script[nonce]')?.getAttribute('nonce') || undefined;
|
|
180
|
-
```
|
|
181
|
-
|
|
182
|
-
The nonce must be generated on the server for each request. The package NEVER generates a nonce in the browser.
|
|
183
|
-
|
|
184
|
-
## CSP Compliance (Important)
|
|
153
|
+
## CSP guide
|
|
185
154
|
|
|
186
|
-
|
|
155
|
+
If your app has no strict CSP, default integration works without extra setup.
|
|
187
156
|
|
|
188
|
-
|
|
157
|
+
If your app uses strict nonce-based CSP:
|
|
189
158
|
|
|
190
|
-
- Generate
|
|
191
|
-
- Use
|
|
192
|
-
-
|
|
193
|
-
-
|
|
194
|
-
-
|
|
195
|
-
-
|
|
159
|
+
- Generate nonce on server for each request.
|
|
160
|
+
- Use same nonce in:
|
|
161
|
+
- `Content-Security-Policy` header
|
|
162
|
+
- script tags loading your Angular app
|
|
163
|
+
- `cspNonce` option/input
|
|
164
|
+
- Do not generate nonce in browser.
|
|
196
165
|
|
|
197
|
-
###
|
|
166
|
+
### Strict policy example (recommended for security)
|
|
198
167
|
|
|
199
|
-
|
|
200
|
-
|
|
201
|
-
|
|
202
|
-
|
|
203
|
-
|
|
204
|
-
|
|
205
|
-
|
|
206
|
-
|
|
207
|
-
style-src 'self' https://embed.mycybersiara.com https://mycybersiara.com https://fonts.googleapis.com https://cdnjs.cloudflare.com 'unsafe-inline';
|
|
208
|
-
font-src 'self' https://fonts.gstatic.com https://mycybersiara.com https://cdnjs.cloudflare.com data:;
|
|
209
|
-
img-src 'self' data: https://embed.mycybersiara.com https://embedcdn.mycybersiara.com https://mycybersiara.com;
|
|
210
|
-
">
|
|
211
|
-
|
|
212
|
-
<script nonce="SIARASHIELD_NONCE" src="https://embedcdn.mycybersiara.com/capcha-temple/js/jquery.min.js"></script>
|
|
213
|
-
<script nonce="SIARASHIELD_NONCE" src="https://embedcdn.mycybersiara.com/CaptchaFormate/CaptchaResources.js"></script>
|
|
168
|
+
```http
|
|
169
|
+
default-src 'self';
|
|
170
|
+
script-src 'self' 'nonce-<dynamic>' https://embed.mycybersiara.com https://embedcdn.mycybersiara.com;
|
|
171
|
+
script-src-elem 'self' 'nonce-<dynamic>' https://embed.mycybersiara.com https://embedcdn.mycybersiara.com;
|
|
172
|
+
connect-src 'self' https://embed.mycybersiara.com https://embedcdn.mycybersiara.com;
|
|
173
|
+
style-src 'self' https://embed.mycybersiara.com https://mycybersiara.com https://fonts.googleapis.com https://cdnjs.cloudflare.com;
|
|
174
|
+
font-src 'self' https://fonts.gstatic.com https://mycybersiara.com https://cdnjs.cloudflare.com data:;
|
|
175
|
+
img-src 'self' data: https://embed.mycybersiara.com https://embedcdn.mycybersiara.com https://mycybersiara.com;
|
|
214
176
|
```
|
|
215
177
|
|
|
216
|
-
|
|
178
|
+
### Compatibility policy (easier integration, weaker security)
|
|
179
|
+
|
|
180
|
+
Use only when customer policy allows `'unsafe-inline'`:
|
|
217
181
|
|
|
218
182
|
```ts
|
|
219
183
|
import { getSiaraShieldCspPolicy } from 'siarashield_workspace';
|
|
@@ -224,107 +188,53 @@ const csp = getSiaraShieldCspPolicy({
|
|
|
224
188
|
});
|
|
225
189
|
```
|
|
226
190
|
|
|
227
|
-
|
|
191
|
+
## Script loading behavior
|
|
228
192
|
|
|
229
|
-
|
|
230
|
-
|
|
231
|
-
```html
|
|
232
|
-
<script nonce="siarashield-example-nonce">
|
|
233
|
-
window.__cspNonce = "siarashield-example-nonce";
|
|
234
|
-
</script>
|
|
235
|
-
<script nonce="siarashield-example-nonce" src="https://embedcdn.mycybersiara.com/capcha-temple/js/jquery.min.js"></script>
|
|
236
|
-
<script nonce="siarashield-example-nonce" src="https://embedcdn.mycybersiara.com/CaptchaFormate/CaptchaResources.js"></script>
|
|
237
|
-
|
|
238
|
-
<meta http-equiv="Content-Security-Policy" content="
|
|
239
|
-
default-src 'self';
|
|
240
|
-
script-src 'self' 'nonce-siarashield-example-nonce' https://embed.mycybersiara.com https://embedcdn.mycybersiara.com;
|
|
241
|
-
script-src-elem 'self' 'nonce-siarashield-example-nonce' https://embed.mycybersiara.com https://embedcdn.mycybersiara.com;
|
|
242
|
-
connect-src 'self' https://embed.mycybersiara.com https://embedcdn.mycybersiara.com;
|
|
243
|
-
style-src 'self' https://embed.mycybersiara.com https://mycybersiara.com https://fonts.googleapis.com https://cdnjs.cloudflare.com 'unsafe-inline';
|
|
244
|
-
font-src 'self' https://fonts.gstatic.com https://mycybersiara.com https://cdnjs.cloudflare.com data:;
|
|
245
|
-
img-src 'self' data: https://embed.mycybersiara.com https://embedcdn.mycybersiara.com https://mycybersiara.com;
|
|
246
|
-
">
|
|
247
|
-
```
|
|
248
|
-
|
|
249
|
-
### Strict production CSP policy (advanced)
|
|
250
|
-
|
|
251
|
-
In production, send CSP as an HTTP response header and inject the same nonce value into script tags and `initSiaraShield({ cspNonce })`.
|
|
252
|
-
|
|
253
|
-
```http
|
|
254
|
-
default-src 'self';
|
|
255
|
-
script-src 'self' 'nonce-<dynamic>' https://embed.mycybersiara.com https://embedcdn.mycybersiara.com;
|
|
256
|
-
script-src-elem 'self' 'nonce-<dynamic>' https://embed.mycybersiara.com https://embedcdn.mycybersiara.com;
|
|
257
|
-
connect-src 'self' https://embed.mycybersiara.com https://embedcdn.mycybersiara.com;
|
|
258
|
-
style-src 'self' https://embed.mycybersiara.com https://mycybersiara.com https://fonts.googleapis.com https://cdnjs.cloudflare.com;
|
|
259
|
-
font-src 'self' https://fonts.gstatic.com https://mycybersiara.com https://cdnjs.cloudflare.com data:;
|
|
260
|
-
img-src 'self' data: https://embed.mycybersiara.com https://embedcdn.mycybersiara.com https://mycybersiara.com;
|
|
261
|
-
```
|
|
262
|
-
|
|
263
|
-
Example with the same server-generated nonce in script tags:
|
|
264
|
-
|
|
265
|
-
```html
|
|
266
|
-
<script nonce="SIARASHIELD_NONCE">
|
|
267
|
-
window.__cspNonce = "SIARASHIELD_NONCE";
|
|
268
|
-
</script>
|
|
269
|
-
<script nonce="SIARASHIELD_NONCE" src="https://embedcdn.mycybersiara.com/capcha-temple/js/jquery.min.js"></script>
|
|
270
|
-
<script nonce="SIARASHIELD_NONCE" src="https://embedcdn.mycybersiara.com/CaptchaFormate/CaptchaResources.js"></script>
|
|
271
|
-
```
|
|
193
|
+
By default (`loadJQuery: true`) package loads required vendor resources automatically:
|
|
272
194
|
|
|
273
|
-
|
|
195
|
+
- `https://embedcdn.mycybersiara.com/capcha-temple/js/jquery.min.js` (fallback when jQuery not already present)
|
|
196
|
+
- `https://embedcdn.mycybersiara.com/CaptchaFormate/CaptchaResources.js`
|
|
197
|
+
- `https://embed.mycybersiara.com/CaptchaFormate/SiaraShield_Validation.js`
|
|
274
198
|
|
|
275
|
-
|
|
199
|
+
Most users should **not** add these script tags manually.
|
|
276
200
|
|
|
277
|
-
|
|
201
|
+
If your app already has jQuery, set `[loadJQuery]="false"` (or `loadJQuery: false` in function API).
|
|
278
202
|
|
|
279
|
-
|
|
280
|
-
- For easiest client integration, use the compatibility policy above (`includeUnsafeInlineScript: true`, `includeUnsafeInlineStyle: true`).
|
|
281
|
-
- Generating a new nonce in the browser (or per function call) does not help. CSP nonce must match the value from server response headers for that page load.
|
|
203
|
+
## Advanced validation options
|
|
282
204
|
|
|
283
|
-
|
|
205
|
+
`checkCaptchaAsync(...)` and `checkSiaraShieldCaptchaAsync(...)` support optional tuning:
|
|
284
206
|
|
|
285
|
-
|
|
207
|
+
- `timeoutMs`
|
|
208
|
+
- `pollIntervalMs`
|
|
209
|
+
- `beforeCheckDelayMs`
|
|
210
|
+
- `retryOnFalseMs` (default `0`)
|
|
211
|
+
- `falseResultTokenWaitMs` (default `900`)
|
|
286
212
|
|
|
287
|
-
|
|
288
|
-
- If jQuery is not already available, the package loads:
|
|
289
|
-
- `https://embedcdn.mycybersiara.com/capcha-temple/js/jquery.min.js`
|
|
290
|
-
- The package also loads:
|
|
291
|
-
- `https://embedcdn.mycybersiara.com/CaptchaFormate/CaptchaResources.js`
|
|
292
|
-
- `https://embed.mycybersiara.com/CaptchaFormate/SiaraShield_Validation.js`
|
|
293
|
-
- jQuery and the captcha bootstrap process may create script elements dynamically (including via HTML strings / DOM fragments).
|
|
294
|
-
- When `cspNonce` is provided, dynamically inserted `<script>` elements are assigned that nonce (including scripts inserted indirectly through DOM manipulation).
|
|
295
|
-
- The plugin includes a built-in guard that blocks vendor synthetic clicks on `.CaptchaSubmit` and suppresses rapid duplicate user clicks.
|
|
296
|
-
- No application code changes are required for this guard.
|
|
297
|
-
- This is required for compatibility with strict nonce-based CSP policies.
|
|
298
|
-
- If a valid CSP nonce is required but not passed, browsers will block those script loads and the captcha will fail to initialize.
|
|
299
|
-
- If your app already loads jQuery, set `loadJQuery: false`.
|
|
300
|
-
- If your app loads jQuery from another CDN, allow that host in `script-src` and `script-src-elem`.
|
|
213
|
+
Default behavior is tuned to avoid requiring a second user click when token arrives slightly after first response.
|
|
301
214
|
|
|
302
215
|
## Quick troubleshooting
|
|
303
216
|
|
|
304
217
|
- Captcha not visible:
|
|
305
|
-
-
|
|
306
|
-
-
|
|
307
|
-
-
|
|
218
|
+
- Confirm component is rendered and not hidden.
|
|
219
|
+
- Confirm submit button has `class="CaptchaSubmit"`.
|
|
220
|
+
- For function API, confirm `<div class="SiaraShield"></div>` exists.
|
|
308
221
|
- `CheckCaptcha` not available:
|
|
309
|
-
- Ensure initialization completed
|
|
310
|
-
- Ensure CSP allows
|
|
311
|
-
-
|
|
312
|
-
-
|
|
313
|
-
|
|
314
|
-
-
|
|
315
|
-
|
|
316
|
-
|
|
317
|
-
|
|
318
|
-
|
|
319
|
-
|
|
320
|
-
|
|
321
|
-
-
|
|
322
|
-
-
|
|
323
|
-
|
|
324
|
-
|
|
325
|
-
- Captcha check runs before API call.
|
|
326
|
-
|
|
327
|
-
## Build and pack (library maintainers)
|
|
222
|
+
- Ensure initialization completed.
|
|
223
|
+
- Ensure CSP allows required hosts and nonce.
|
|
224
|
+
- Token not available immediately:
|
|
225
|
+
- Use async check methods (`checkCaptchaAsync` / `checkSiaraShieldCaptchaAsync`).
|
|
226
|
+
- CSP errors:
|
|
227
|
+
- Confirm same server-generated nonce in header, script tags, and `cspNonce`.
|
|
228
|
+
|
|
229
|
+
## Final checklist
|
|
230
|
+
|
|
231
|
+
- Public key in frontend, private key only in backend.
|
|
232
|
+
- Submit button contains `CaptchaSubmit`.
|
|
233
|
+
- Captcha check runs before submit API call.
|
|
234
|
+
- Only one integration style is used per page.
|
|
235
|
+
- CSP nonce is wired correctly for strict CSP deployments.
|
|
236
|
+
|
|
237
|
+
## Build and pack (maintainers)
|
|
328
238
|
|
|
329
239
|
```bash
|
|
330
240
|
npm run build:lib
|
|
@@ -185,7 +185,8 @@ export function prepareScriptNonce(documentRef, explicitNonce) {
|
|
|
185
185
|
return resolvedNonce;
|
|
186
186
|
}
|
|
187
187
|
export function loadScript(documentRef, src, options) {
|
|
188
|
-
|
|
188
|
+
// CSP nonce propagation is intentionally disabled in plugin runtime path for compatibility testing.
|
|
189
|
+
const nonce = normalizeNonce(options?.nonce);
|
|
189
190
|
const statusBySrc = getStatusBySrc(documentRef);
|
|
190
191
|
const pendingBySrc = getPendingBySrc(documentRef);
|
|
191
192
|
const existing = documentRef.querySelector(`script[src="${src}"]`);
|
|
@@ -222,4 +223,4 @@ export function loadScript(documentRef, src, options) {
|
|
|
222
223
|
pendingBySrc.set(src, pending);
|
|
223
224
|
return pending;
|
|
224
225
|
}
|
|
225
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"siara-shield-script-utils.js","sourceRoot":"","sources":["../../../../projects/siarashield-workspace/src/lib/siara-shield-script-utils.ts"],"names":[],"mappings":"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","sourcesContent":["type ScriptStatus = 'loading' | 'loaded' | 'error';\n\nexport interface ScriptLoadOptions {\n  nonce?: string;\n}\n\ninterface DynamicScriptNoncePatchState {\n  nonceByDocument: WeakMap<Document, string>;\n  observerByDocument: WeakMap<Document, MutationObserver>;\n  inlineCaptchaScriptByDocument: WeakMap<Document, Set<string>>;\n  patched: boolean;\n}\n\nconst SCRIPT_STATUS_BY_DOCUMENT = new WeakMap<Document, Map<string, ScriptStatus>>();\nconst SCRIPT_PENDING_BY_DOCUMENT = new WeakMap<Document, Map<string, Promise<void>>>();\nconst DYNAMIC_SCRIPT_NONCE_STATE_KEY = '__siaraShieldDynamicScriptNonceState__';\n\nfunction getStatusBySrc(documentRef: Document): Map<string, ScriptStatus> {\n  let statusBySrc = SCRIPT_STATUS_BY_DOCUMENT.get(documentRef);\n  if (!statusBySrc) {\n    statusBySrc = new Map<string, ScriptStatus>();\n    SCRIPT_STATUS_BY_DOCUMENT.set(documentRef, statusBySrc);\n  }\n\n  return statusBySrc;\n}\n\nfunction getPendingBySrc(documentRef: Document): Map<string, Promise<void>> {\n  let pendingBySrc = SCRIPT_PENDING_BY_DOCUMENT.get(documentRef);\n  if (!pendingBySrc) {\n    pendingBySrc = new Map<string, Promise<void>>();\n    SCRIPT_PENDING_BY_DOCUMENT.set(documentRef, pendingBySrc);\n  }\n\n  return pendingBySrc;\n}\n\nfunction getDynamicScriptNoncePatchState(): DynamicScriptNoncePatchState {\n  const globalState = globalThis as typeof globalThis & {\n    [DYNAMIC_SCRIPT_NONCE_STATE_KEY]?: DynamicScriptNoncePatchState;\n  };\n\n  if (!globalState[DYNAMIC_SCRIPT_NONCE_STATE_KEY]) {\n    globalState[DYNAMIC_SCRIPT_NONCE_STATE_KEY] = {\n      nonceByDocument: new WeakMap<Document, string>(),\n      observerByDocument: new WeakMap<Document, MutationObserver>(),\n      inlineCaptchaScriptByDocument: new WeakMap<Document, Set<string>>(),\n      patched: false,\n    };\n  }\n\n  return globalState[DYNAMIC_SCRIPT_NONCE_STATE_KEY];\n}\n\nfunction normalizeNonce(nonce?: string | null): string | undefined {\n  const trimmed = nonce?.trim();\n  return trimmed ? trimmed : undefined;\n}\n\nfunction isScriptElement(node: Node): node is HTMLScriptElement {\n  return node instanceof Element && node.tagName.toLowerCase() === 'script';\n}\n\nexport function applyScriptNonce(script: HTMLScriptElement, nonce?: string): void {\n  const resolvedNonce = normalizeNonce(nonce);\n  if (!resolvedNonce) {\n    return;\n  }\n\n  script.setAttribute('nonce', resolvedNonce);\n  script.nonce = resolvedNonce;\n}\n\nfunction applyTrackedNonce(node: Node): void {\n  if (!isScriptElement(node)) {\n    return;\n  }\n\n  const documentRef = node.ownerDocument;\n  if (!documentRef) {\n    return;\n  }\n\n  const nonce = getDynamicScriptNoncePatchState().nonceByDocument.get(documentRef);\n  applyScriptNonce(node, nonce);\n}\n\nfunction getTrackedInlineCaptchaScripts(documentRef: Document): Set<string> {\n  const patchState = getDynamicScriptNoncePatchState();\n  let trackedScripts = patchState.inlineCaptchaScriptByDocument.get(documentRef);\n  if (!trackedScripts) {\n    trackedScripts = new Set<string>();\n    patchState.inlineCaptchaScriptByDocument.set(documentRef, trackedScripts);\n  }\n\n  return trackedScripts;\n}\n\nfunction shouldSkipDuplicateInlineCaptchaScript(node: Node): boolean {\n  if (!isScriptElement(node) || Boolean(node.src)) {\n    return false;\n  }\n\n  const scriptBody = (node.textContent ?? '').trim();\n  if (!scriptBody || !scriptBody.includes('currentLangCode')) {\n    return false;\n  }\n\n  const documentRef = node.ownerDocument;\n  if (!documentRef) {\n    return false;\n  }\n\n  const trackedScripts = getTrackedInlineCaptchaScripts(documentRef);\n  if (trackedScripts.has(scriptBody)) {\n    return true;\n  }\n\n  trackedScripts.add(scriptBody);\n  return false;\n}\n\nfunction walkAndApplyNonce(root: Node, nonce?: string): void {\n  if (!nonce) return;\n\n  // Fast path: root itself is a script.\n  if (isScriptElement(root)) {\n    applyScriptNonce(root, nonce);\n    return;\n  }\n\n  // Common path for jQuery: append DocumentFragment containing scripts created via HTML parsing.\n  if (root instanceof Element || root instanceof DocumentFragment) {\n    const scripts = root.querySelectorAll?.('script') ?? [];\n    for (const script of scripts) {\n      applyScriptNonce(script as HTMLScriptElement, nonce);\n    }\n  }\n}\n\nfunction ensureNonceMutationObserver(documentRef: Document): void {\n  const patchState = getDynamicScriptNoncePatchState();\n  if (patchState.observerByDocument.get(documentRef)) {\n    return;\n  }\n\n  const observer = new MutationObserver((records) => {\n    const nonce = patchState.nonceByDocument.get(documentRef);\n    if (!nonce) return;\n\n    for (const record of records) {\n      for (const node of record.addedNodes) {\n        walkAndApplyNonce(node, nonce);\n      }\n    }\n  });\n\n  // Observe the full document because scripts can be injected into body/head by vendor/jQuery.\n  const root = documentRef.documentElement ?? documentRef;\n  observer.observe(root, { childList: true, subtree: true });\n  patchState.observerByDocument.set(documentRef, observer);\n}\n\nfunction teardownNonceMutationObserver(documentRef: Document): void {\n  const patchState = getDynamicScriptNoncePatchState();\n  const observer = patchState.observerByDocument.get(documentRef);\n  if (!observer) return;\n  observer.disconnect();\n  patchState.observerByDocument.delete(documentRef);\n}\n\nfunction patchDynamicScriptInsertion(): void {\n  const patchState = getDynamicScriptNoncePatchState();\n  if (patchState.patched) {\n    return;\n  }\n\n  const originalAppendChild = Node.prototype.appendChild;\n  const originalInsertBefore = Node.prototype.insertBefore;\n  const originalReplaceChild = Node.prototype.replaceChild;\n\n  Node.prototype.appendChild = function <T extends Node>(node: T): T {\n    // Vendor captcha injects an inline snippet with `currentLangCode` that crashes when appended twice.\n    if (shouldSkipDuplicateInlineCaptchaScript(node)) {\n      return node;\n    }\n    applyTrackedNonce(node);\n    return originalAppendChild.call(this, node) as T;\n  };\n\n  Node.prototype.insertBefore = function <T extends Node>(node: T, child: Node | null): T {\n    if (shouldSkipDuplicateInlineCaptchaScript(node)) {\n      return node;\n    }\n    applyTrackedNonce(node);\n    return originalInsertBefore.call(this, node, child) as T;\n  };\n\n  Node.prototype.replaceChild = function <T extends Node>(node: Node, child: T): T {\n    applyTrackedNonce(node);\n    return originalReplaceChild.call(this, node, child) as T;\n  };\n\n  patchState.patched = true;\n}\n\nexport function resolveCspNonce(documentRef: Document, explicitNonce?: string): string | undefined {\n  const resolvedExplicitNonce = normalizeNonce(explicitNonce);\n  if (resolvedExplicitNonce) {\n    return resolvedExplicitNonce;\n  }\n\n  const scriptWithNonce = documentRef.querySelector<HTMLScriptElement>('script[nonce]');\n  const nonceFromScript = normalizeNonce(scriptWithNonce?.getAttribute('nonce') ?? scriptWithNonce?.nonce);\n  if (nonceFromScript) {\n    return nonceFromScript;\n  }\n\n  const nonceMeta = documentRef.querySelector<HTMLMetaElement>('meta[name=\"csp-nonce\"]');\n  return normalizeNonce(nonceMeta?.content);\n}\n\nexport function prepareScriptNonce(documentRef: Document, explicitNonce?: string): string | undefined {\n  const resolvedNonce = resolveCspNonce(documentRef, explicitNonce);\n  const patchState = getDynamicScriptNoncePatchState();\n\n  if (resolvedNonce) {\n    // DOM patching is only needed when a CSP nonce must be propagated.\n    patchDynamicScriptInsertion();\n    patchState.nonceByDocument.set(documentRef, resolvedNonce);\n    ensureNonceMutationObserver(documentRef);\n  } else {\n    patchState.nonceByDocument.delete(documentRef);\n    teardownNonceMutationObserver(documentRef);\n  }\n\n  return resolvedNonce;\n}\n\nexport function loadScript(documentRef: Document, src: string, options?: ScriptLoadOptions): Promise<void> {\n  const nonce = prepareScriptNonce(documentRef, options?.nonce);\n  const statusBySrc = getStatusBySrc(documentRef);\n  const pendingBySrc = getPendingBySrc(documentRef);\n  const existing = documentRef.querySelector<HTMLScriptElement>(`script[src=\"${src}\"]`);\n\n  if (existing) {\n    applyScriptNonce(existing, nonce);\n    const status = statusBySrc.get(src);\n    if (status === 'loaded') {\n      return Promise.resolve();\n    }\n\n    const pending = pendingBySrc.get(src);\n    if (pending) {\n      return pending;\n    }\n\n    return Promise.resolve();\n  }\n\n  statusBySrc.set(src, 'loading');\n\n  const pending = new Promise<void>((resolve, reject) => {\n    const script = documentRef.createElement('script');\n    script.src = src;\n    script.async = true;\n    applyScriptNonce(script, nonce);\n    script.onload = () => {\n      statusBySrc.set(src, 'loaded');\n      pendingBySrc.delete(src);\n      resolve();\n    };\n    script.onerror = () => {\n      statusBySrc.set(src, 'error');\n      pendingBySrc.delete(src);\n      reject(new Error(`Failed to load script: ${src}. Check CSP allowlist and nonce configuration.`));\n    };\n    documentRef.head.appendChild(script);\n  });\n\n  pendingBySrc.set(src, pending);\n  return pending;\n}\n"]}
|
|
226
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"siara-shield-script-utils.js","sourceRoot":"","sources":["../../../../projects/siarashield-workspace/src/lib/siara-shield-script-utils.ts"],"names":[],"mappings":"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","sourcesContent":["type ScriptStatus = 'loading' | 'loaded' | 'error';\n\nexport interface ScriptLoadOptions {\n  nonce?: string;\n}\n\ninterface DynamicScriptNoncePatchState {\n  nonceByDocument: WeakMap<Document, string>;\n  observerByDocument: WeakMap<Document, MutationObserver>;\n  inlineCaptchaScriptByDocument: WeakMap<Document, Set<string>>;\n  patched: boolean;\n}\n\nconst SCRIPT_STATUS_BY_DOCUMENT = new WeakMap<Document, Map<string, ScriptStatus>>();\nconst SCRIPT_PENDING_BY_DOCUMENT = new WeakMap<Document, Map<string, Promise<void>>>();\nconst DYNAMIC_SCRIPT_NONCE_STATE_KEY = '__siaraShieldDynamicScriptNonceState__';\n\nfunction getStatusBySrc(documentRef: Document): Map<string, ScriptStatus> {\n  let statusBySrc = SCRIPT_STATUS_BY_DOCUMENT.get(documentRef);\n  if (!statusBySrc) {\n    statusBySrc = new Map<string, ScriptStatus>();\n    SCRIPT_STATUS_BY_DOCUMENT.set(documentRef, statusBySrc);\n  }\n\n  return statusBySrc;\n}\n\nfunction getPendingBySrc(documentRef: Document): Map<string, Promise<void>> {\n  let pendingBySrc = SCRIPT_PENDING_BY_DOCUMENT.get(documentRef);\n  if (!pendingBySrc) {\n    pendingBySrc = new Map<string, Promise<void>>();\n    SCRIPT_PENDING_BY_DOCUMENT.set(documentRef, pendingBySrc);\n  }\n\n  return pendingBySrc;\n}\n\nfunction getDynamicScriptNoncePatchState(): DynamicScriptNoncePatchState {\n  const globalState = globalThis as typeof globalThis & {\n    [DYNAMIC_SCRIPT_NONCE_STATE_KEY]?: DynamicScriptNoncePatchState;\n  };\n\n  if (!globalState[DYNAMIC_SCRIPT_NONCE_STATE_KEY]) {\n    globalState[DYNAMIC_SCRIPT_NONCE_STATE_KEY] = {\n      nonceByDocument: new WeakMap<Document, string>(),\n      observerByDocument: new WeakMap<Document, MutationObserver>(),\n      inlineCaptchaScriptByDocument: new WeakMap<Document, Set<string>>(),\n      patched: false,\n    };\n  }\n\n  return globalState[DYNAMIC_SCRIPT_NONCE_STATE_KEY];\n}\n\nfunction normalizeNonce(nonce?: string | null): string | undefined {\n  const trimmed = nonce?.trim();\n  return trimmed ? trimmed : undefined;\n}\n\nfunction isScriptElement(node: Node): node is HTMLScriptElement {\n  return node instanceof Element && node.tagName.toLowerCase() === 'script';\n}\n\nexport function applyScriptNonce(script: HTMLScriptElement, nonce?: string): void {\n  const resolvedNonce = normalizeNonce(nonce);\n  if (!resolvedNonce) {\n    return;\n  }\n\n  script.setAttribute('nonce', resolvedNonce);\n  script.nonce = resolvedNonce;\n}\n\nfunction applyTrackedNonce(node: Node): void {\n  if (!isScriptElement(node)) {\n    return;\n  }\n\n  const documentRef = node.ownerDocument;\n  if (!documentRef) {\n    return;\n  }\n\n  const nonce = getDynamicScriptNoncePatchState().nonceByDocument.get(documentRef);\n  applyScriptNonce(node, nonce);\n}\n\nfunction getTrackedInlineCaptchaScripts(documentRef: Document): Set<string> {\n  const patchState = getDynamicScriptNoncePatchState();\n  let trackedScripts = patchState.inlineCaptchaScriptByDocument.get(documentRef);\n  if (!trackedScripts) {\n    trackedScripts = new Set<string>();\n    patchState.inlineCaptchaScriptByDocument.set(documentRef, trackedScripts);\n  }\n\n  return trackedScripts;\n}\n\nfunction shouldSkipDuplicateInlineCaptchaScript(node: Node): boolean {\n  if (!isScriptElement(node) || Boolean(node.src)) {\n    return false;\n  }\n\n  const scriptBody = (node.textContent ?? '').trim();\n  if (!scriptBody || !scriptBody.includes('currentLangCode')) {\n    return false;\n  }\n\n  const documentRef = node.ownerDocument;\n  if (!documentRef) {\n    return false;\n  }\n\n  const trackedScripts = getTrackedInlineCaptchaScripts(documentRef);\n  if (trackedScripts.has(scriptBody)) {\n    return true;\n  }\n\n  trackedScripts.add(scriptBody);\n  return false;\n}\n\nfunction walkAndApplyNonce(root: Node, nonce?: string): void {\n  if (!nonce) return;\n\n  // Fast path: root itself is a script.\n  if (isScriptElement(root)) {\n    applyScriptNonce(root, nonce);\n    return;\n  }\n\n  // Common path for jQuery: append DocumentFragment containing scripts created via HTML parsing.\n  if (root instanceof Element || root instanceof DocumentFragment) {\n    const scripts = root.querySelectorAll?.('script') ?? [];\n    for (const script of scripts) {\n      applyScriptNonce(script as HTMLScriptElement, nonce);\n    }\n  }\n}\n\nfunction ensureNonceMutationObserver(documentRef: Document): void {\n  const patchState = getDynamicScriptNoncePatchState();\n  if (patchState.observerByDocument.get(documentRef)) {\n    return;\n  }\n\n  const observer = new MutationObserver((records) => {\n    const nonce = patchState.nonceByDocument.get(documentRef);\n    if (!nonce) return;\n\n    for (const record of records) {\n      for (const node of record.addedNodes) {\n        walkAndApplyNonce(node, nonce);\n      }\n    }\n  });\n\n  // Observe the full document because scripts can be injected into body/head by vendor/jQuery.\n  const root = documentRef.documentElement ?? documentRef;\n  observer.observe(root, { childList: true, subtree: true });\n  patchState.observerByDocument.set(documentRef, observer);\n}\n\nfunction teardownNonceMutationObserver(documentRef: Document): void {\n  const patchState = getDynamicScriptNoncePatchState();\n  const observer = patchState.observerByDocument.get(documentRef);\n  if (!observer) return;\n  observer.disconnect();\n  patchState.observerByDocument.delete(documentRef);\n}\n\nfunction patchDynamicScriptInsertion(): void {\n  const patchState = getDynamicScriptNoncePatchState();\n  if (patchState.patched) {\n    return;\n  }\n\n  const originalAppendChild = Node.prototype.appendChild;\n  const originalInsertBefore = Node.prototype.insertBefore;\n  const originalReplaceChild = Node.prototype.replaceChild;\n\n  Node.prototype.appendChild = function <T extends Node>(node: T): T {\n    // Vendor captcha injects an inline snippet with `currentLangCode` that crashes when appended twice.\n    if (shouldSkipDuplicateInlineCaptchaScript(node)) {\n      return node;\n    }\n    applyTrackedNonce(node);\n    return originalAppendChild.call(this, node) as T;\n  };\n\n  Node.prototype.insertBefore = function <T extends Node>(node: T, child: Node | null): T {\n    if (shouldSkipDuplicateInlineCaptchaScript(node)) {\n      return node;\n    }\n    applyTrackedNonce(node);\n    return originalInsertBefore.call(this, node, child) as T;\n  };\n\n  Node.prototype.replaceChild = function <T extends Node>(node: Node, child: T): T {\n    applyTrackedNonce(node);\n    return originalReplaceChild.call(this, node, child) as T;\n  };\n\n  patchState.patched = true;\n}\n\nexport function resolveCspNonce(documentRef: Document, explicitNonce?: string): string | undefined {\n  const resolvedExplicitNonce = normalizeNonce(explicitNonce);\n  if (resolvedExplicitNonce) {\n    return resolvedExplicitNonce;\n  }\n\n  const scriptWithNonce = documentRef.querySelector<HTMLScriptElement>('script[nonce]');\n  const nonceFromScript = normalizeNonce(scriptWithNonce?.getAttribute('nonce') ?? scriptWithNonce?.nonce);\n  if (nonceFromScript) {\n    return nonceFromScript;\n  }\n\n  const nonceMeta = documentRef.querySelector<HTMLMetaElement>('meta[name=\"csp-nonce\"]');\n  return normalizeNonce(nonceMeta?.content);\n}\n\nexport function prepareScriptNonce(documentRef: Document, explicitNonce?: string): string | undefined {\n  const resolvedNonce = resolveCspNonce(documentRef, explicitNonce);\n  const patchState = getDynamicScriptNoncePatchState();\n\n  if (resolvedNonce) {\n    // DOM patching is only needed when a CSP nonce must be propagated.\n    patchDynamicScriptInsertion();\n    patchState.nonceByDocument.set(documentRef, resolvedNonce);\n    ensureNonceMutationObserver(documentRef);\n  } else {\n    patchState.nonceByDocument.delete(documentRef);\n    teardownNonceMutationObserver(documentRef);\n  }\n\n  return resolvedNonce;\n}\n\nexport function loadScript(documentRef: Document, src: string, options?: ScriptLoadOptions): Promise<void> {\n  // CSP nonce propagation is intentionally disabled in plugin runtime path for compatibility testing.\n  const nonce = normalizeNonce(options?.nonce);\n  const statusBySrc = getStatusBySrc(documentRef);\n  const pendingBySrc = getPendingBySrc(documentRef);\n  const existing = documentRef.querySelector<HTMLScriptElement>(`script[src=\"${src}\"]`);\n\n  if (existing) {\n    applyScriptNonce(existing, nonce);\n    const status = statusBySrc.get(src);\n    if (status === 'loaded') {\n      return Promise.resolve();\n    }\n\n    const pending = pendingBySrc.get(src);\n    if (pending) {\n      return pending;\n    }\n\n    return Promise.resolve();\n  }\n\n  statusBySrc.set(src, 'loading');\n\n  const pending = new Promise<void>((resolve, reject) => {\n    const script = documentRef.createElement('script');\n    script.src = src;\n    script.async = true;\n    applyScriptNonce(script, nonce);\n    script.onload = () => {\n      statusBySrc.set(src, 'loaded');\n      pendingBySrc.delete(src);\n      resolve();\n    };\n    script.onerror = () => {\n      statusBySrc.set(src, 'error');\n      pendingBySrc.delete(src);\n      reject(new Error(`Failed to load script: ${src}. Check CSP allowlist and nonce configuration.`));\n    };\n    documentRef.head.appendChild(script);\n  });\n\n  pendingBySrc.set(src, pending);\n  return pending;\n}\n"]}
|