1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17 package org.mal_lang.langspec;
18
19 import static java.util.Objects.requireNonNull;
20
21 import jakarta.json.Json;
22 import jakarta.json.JsonObject;
23 import java.util.ArrayList;
24 import java.util.LinkedHashSet;
25 import java.util.List;
26 import java.util.Set;
27 import org.mal_lang.langspec.builders.AttackStepBuilder;
28 import org.mal_lang.langspec.step.StepExpression;
29 import org.mal_lang.langspec.ttc.TtcExpression;
30
31
32
33
34
35
36 public final class AttackStep {
37 private final String name;
38 private final Meta meta;
39 private final Asset asset;
40 private final AttackStepType type;
41 private final Set<String> tags;
42 private final Risk risk;
43 private final TtcExpression ttc;
44 private final Steps requires;
45 private final Steps reaches;
46
47 private AttackStep(
48 String name,
49 Meta meta,
50 Asset asset,
51 AttackStepType type,
52 List<String> tags,
53 Risk risk,
54 TtcExpression ttc,
55 Steps requires,
56 Steps reaches) {
57 this.name = requireNonNull(name);
58 this.meta = requireNonNull(meta);
59 this.asset = requireNonNull(asset);
60 this.type = requireNonNull(type);
61 this.tags = new LinkedHashSet<>(requireNonNull(tags));
62 this.risk = risk;
63 this.ttc = ttc;
64 this.requires = requires;
65 this.reaches = reaches;
66 }
67
68
69
70
71
72
73
74 public String getName() {
75 return this.name;
76 }
77
78
79
80
81
82
83
84 public Meta getMeta() {
85 return this.meta;
86 }
87
88
89
90
91
92
93
94 public Asset getAsset() {
95 return this.asset;
96 }
97
98
99
100
101
102
103
104 public AttackStepType getType() {
105 return this.type;
106 }
107
108
109
110
111
112
113
114
115
116 public boolean hasLocalTag(String name) {
117 return this.tags.contains(requireNonNull(name));
118 }
119
120
121
122
123
124
125
126 public List<String> getLocalTags() {
127 return List.copyOf(this.tags);
128 }
129
130
131
132
133
134
135
136
137
138 public boolean hasTag(String name) {
139 return this.hasLocalTag(name)
140 || this.hasSuperAttackStep() && this.getSuperAttackStep().hasTag(name);
141 }
142
143
144
145
146
147
148
149 public List<String> getTags() {
150 return List.copyOf(this.getTagsSet());
151 }
152
153 private Set<String> getTagsSet() {
154 var tagsSet =
155 this.hasSuperAttackStep()
156 ? this.getSuperAttackStep().getTagsSet()
157 : new LinkedHashSet<String>();
158 tagsSet.addAll(this.tags);
159 return tagsSet;
160 }
161
162
163
164
165
166
167
168 public boolean hasLocalRisk() {
169 return this.risk != null;
170 }
171
172
173
174
175
176
177
178
179
180 public Risk getLocalRisk() {
181 if (!this.hasLocalRisk()) {
182 throw new UnsupportedOperationException("Local risk not found");
183 }
184 return this.risk;
185 }
186
187
188
189
190
191
192
193 public boolean hasRisk() {
194 return this.hasLocalRisk() || this.hasSuperAttackStep() && this.getSuperAttackStep().hasRisk();
195 }
196
197
198
199
200
201
202
203
204
205 public Risk getRisk() {
206 if (!this.hasRisk()) {
207 throw new UnsupportedOperationException("Risk not found");
208 }
209 return this.hasLocalRisk() ? this.getLocalRisk() : this.getSuperAttackStep().getRisk();
210 }
211
212
213
214
215
216
217
218 public boolean hasLocalTtc() {
219 return this.ttc != null;
220 }
221
222
223
224
225
226
227
228
229
230 public TtcExpression getLocalTtc() {
231 if (!this.hasLocalTtc()) {
232 throw new UnsupportedOperationException("Local TTC not found");
233 }
234 return this.ttc;
235 }
236
237
238
239
240
241
242
243 public boolean hasTtc() {
244 return this.hasLocalTtc() || this.hasSuperAttackStep() && this.getSuperAttackStep().hasTtc();
245 }
246
247
248
249
250
251
252
253
254
255 public TtcExpression getTtc() {
256 if (!this.hasTtc()) {
257 throw new UnsupportedOperationException("TTC not found");
258 }
259 return this.hasLocalTtc() ? this.getLocalTtc() : this.getSuperAttackStep().getTtc();
260 }
261
262
263
264
265
266
267
268 public boolean hasLocalRequires() {
269 return this.requires != null;
270 }
271
272
273
274
275
276
277
278
279
280 public Steps getLocalRequires() {
281 if (!this.hasLocalRequires()) {
282 throw new UnsupportedOperationException("Local requires not found");
283 }
284 return this.requires;
285 }
286
287
288
289
290
291
292
293 public List<StepExpression> getRequires() {
294 return List.copyOf(this.getRequiresList());
295 }
296
297 private List<StepExpression> getRequiresList() {
298 var overrides = this.hasLocalRequires() && this.requires.overrides();
299 var requiresList =
300 this.hasSuperAttackStep() && !overrides
301 ? this.getSuperAttackStep().getRequiresList()
302 : new ArrayList<StepExpression>();
303 if (this.hasLocalRequires()) {
304 requiresList.addAll(this.requires.getStepExpressions());
305 }
306 return requiresList;
307 }
308
309
310
311
312
313
314
315 public boolean hasLocalReaches() {
316 return this.reaches != null;
317 }
318
319
320
321
322
323
324
325
326
327 public Steps getLocalReaches() {
328 if (!this.hasLocalReaches()) {
329 throw new UnsupportedOperationException("Local reaches not found");
330 }
331 return this.reaches;
332 }
333
334
335
336
337
338
339
340 public List<StepExpression> getReaches() {
341 return List.copyOf(this.getReachesList());
342 }
343
344 private List<StepExpression> getReachesList() {
345 var overrides = this.hasLocalReaches() && this.reaches.overrides();
346 var reachesList =
347 this.hasSuperAttackStep() && !overrides
348 ? this.getSuperAttackStep().getReachesList()
349 : new ArrayList<StepExpression>();
350 if (this.hasLocalReaches()) {
351 reachesList.addAll(this.reaches.getStepExpressions());
352 }
353 return reachesList;
354 }
355
356
357
358
359
360
361
362 public boolean hasSuperAttackStep() {
363 return this.asset.hasSuperAsset() && this.asset.getSuperAsset().hasAttackStep(this.name);
364 }
365
366
367
368
369
370
371
372
373
374 public AttackStep getSuperAttackStep() {
375 if (!this.hasSuperAttackStep()) {
376 throw new UnsupportedOperationException("Super attack step not found");
377 }
378 return this.asset.getSuperAsset().getAttackStep(this.name);
379 }
380
381 JsonObject toJson() {
382 var jsonTags = Json.createArrayBuilder();
383 for (var tag : this.tags) {
384 jsonTags.add(tag);
385 }
386
387 var jsonAttackStep =
388 Json.createObjectBuilder()
389 .add("name", this.name)
390 .add("meta", this.meta.toJson())
391 .add("type", this.type.toString())
392 .add("tags", jsonTags);
393 if (this.risk == null) {
394 jsonAttackStep.addNull("risk");
395 } else {
396 jsonAttackStep.add("risk", this.risk.toJson());
397 }
398 if (this.ttc == null) {
399 jsonAttackStep.addNull("ttc");
400 } else {
401 jsonAttackStep.add("ttc", this.ttc.toJson());
402 }
403 if (this.requires == null) {
404 jsonAttackStep.addNull("requires");
405 } else {
406 jsonAttackStep.add("requires", this.requires.toJson());
407 }
408 if (this.reaches == null) {
409 jsonAttackStep.addNull("reaches");
410 } else {
411 jsonAttackStep.add("reaches", this.reaches.toJson());
412 }
413 return jsonAttackStep.build();
414 }
415
416 static AttackStep fromBuilder(AttackStepBuilder builder, Asset asset) {
417 requireNonNull(builder);
418 requireNonNull(asset);
419 var requires = builder.getRequires() == null ? null : Steps.fromBuilder(builder.getRequires());
420 var reaches = builder.getReaches() == null ? null : Steps.fromBuilder(builder.getReaches());
421 return new AttackStep(
422 builder.getName(),
423 Meta.fromBuilder(builder.getMeta()),
424 asset,
425 builder.getType(),
426 builder.getTags(),
427 builder.getRisk(),
428 builder.getTtc(),
429 requires,
430 reaches);
431 }
432 }