View Javadoc
1   /*
2    * Copyright 2020-2022 Foreseeti AB <https://foreseeti.com>
3    *
4    * Licensed under the Apache License, Version 2.0 (the "License");
5    * you may not use this file except in compliance with the License.
6    * You may obtain a copy of the License at
7    *
8    *     http://www.apache.org/licenses/LICENSE-2.0
9    *
10   * Unless required by applicable law or agreed to in writing, software
11   * distributed under the License is distributed on an "AS IS" BASIS,
12   * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13   * See the License for the specific language governing permissions and
14   * limitations under the License.
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   * Immutable class representing an attack step of an asset in a MAL language.
33   *
34   * @since 1.0.0
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     * Returns the name of this {@code AttackStep} object.
70     *
71     * @return the name of this {@code AttackStep} object
72     * @since 1.0.0
73     */
74    public String getName() {
75      return this.name;
76    }
77  
78    /**
79     * Returns the meta info of this {@code AttackStep} object.
80     *
81     * @return the meta info of this {@code AttackStep} object
82     * @since 1.0.0
83     */
84    public Meta getMeta() {
85      return this.meta;
86    }
87  
88    /**
89     * Returns the asset of this {@code AttackStep} object.
90     *
91     * @return the asset of this {@code AttackStep} object
92     * @since 1.0.0
93     */
94    public Asset getAsset() {
95      return this.asset;
96    }
97  
98    /**
99     * Returns the type of this {@code AttackStep} object.
100    *
101    * @return the type of this {@code AttackStep} object
102    * @since 1.0.0
103    */
104   public AttackStepType getType() {
105     return this.type;
106   }
107 
108   /**
109    * Returns whether {@code name} is the name of a local tag in this {@code AttackStep} object.
110    *
111    * @param name the name of the local tag
112    * @return whether {@code name} is the name of a local tag in this {@code AttackStep} object
113    * @throws java.lang.NullPointerException if {@code name} is {@code null}
114    * @since 1.0.0
115    */
116   public boolean hasLocalTag(String name) {
117     return this.tags.contains(requireNonNull(name));
118   }
119 
120   /**
121    * Returns a list of all local tags in this {@code AttackStep} object.
122    *
123    * @return a list of all local tags in this {@code AttackStep} object
124    * @since 1.0.0
125    */
126   public List<String> getLocalTags() {
127     return List.copyOf(this.tags);
128   }
129 
130   /**
131    * Returns whether {@code name} is the name of a tag in this {@code AttackStep} object.
132    *
133    * @param name the name of the tag
134    * @return whether {@code name} is the name of a tag in this {@code AttackStep} object
135    * @throws java.lang.NullPointerException if {@code name} is {@code null}
136    * @since 1.0.0
137    */
138   public boolean hasTag(String name) {
139     return this.hasLocalTag(name)
140         || this.hasSuperAttackStep() && this.getSuperAttackStep().hasTag(name);
141   }
142 
143   /**
144    * Returns a list of all tags in this {@code AttackStep} object.
145    *
146    * @return a list of all tags in this {@code AttackStep} object
147    * @since 1.0.0
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    * Returns whether this {@code AttackStep} object has a local risk.
164    *
165    * @return whether this {@code AttackStep} object has a local risk
166    * @since 1.0.0
167    */
168   public boolean hasLocalRisk() {
169     return this.risk != null;
170   }
171 
172   /**
173    * Returns the local risk of this {@code AttackStep} object.
174    *
175    * @return the local risk of this {@code AttackStep} object
176    * @throws java.lang.UnsupportedOperationException if this {@code AttackStep} object does not have
177    *     a local risk
178    * @since 1.0.0
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    * Returns whether this {@code AttackStep} object has a risk.
189    *
190    * @return whether this {@code AttackStep} object has a risk
191    * @since 1.0.0
192    */
193   public boolean hasRisk() {
194     return this.hasLocalRisk() || this.hasSuperAttackStep() && this.getSuperAttackStep().hasRisk();
195   }
196 
197   /**
198    * Returns the risk of this {@code AttackStep} object.
199    *
200    * @return the risk of this {@code AttackStep} object
201    * @throws java.lang.UnsupportedOperationException if this {@code AttackStep} object does not have
202    *     a risk
203    * @since 1.0.0
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    * Returns whether this {@code AttackStep} object has a local TTC.
214    *
215    * @return whether this {@code AttackStep} object has a local TTC
216    * @since 1.0.0
217    */
218   public boolean hasLocalTtc() {
219     return this.ttc != null;
220   }
221 
222   /**
223    * Returns the local TTC of this {@code AttackStep} object.
224    *
225    * @return the local TTC of this {@code AttackStep} object
226    * @throws java.lang.UnsupportedOperationException if this {@code AttackStep} object does not have
227    *     a local TTC
228    * @since 1.0.0
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    * Returns whether this {@code AttackStep} object has a TTC.
239    *
240    * @return whether this {@code AttackStep} object has a TTC
241    * @since 1.0.0
242    */
243   public boolean hasTtc() {
244     return this.hasLocalTtc() || this.hasSuperAttackStep() && this.getSuperAttackStep().hasTtc();
245   }
246 
247   /**
248    * Returns the TTC of this {@code AttackStep} object.
249    *
250    * @return the TTC of this {@code AttackStep} object
251    * @throws java.lang.UnsupportedOperationException if this {@code AttackStep} object does not have
252    *     a TTC
253    * @since 1.0.0
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    * Returns whether this {@code AttackStep} object has local requires steps.
264    *
265    * @return whether this {@code AttackStep} object has local requires steps
266    * @since 1.0.0
267    */
268   public boolean hasLocalRequires() {
269     return this.requires != null;
270   }
271 
272   /**
273    * Returns the local requires steps of this {@code AttackStep} object.
274    *
275    * @return the local requires steps of this {@code AttackStep} object
276    * @throws java.lang.UnsupportedOperationException if this {@code AttackStep} object does not have
277    *     local requires steps
278    * @since 1.0.0
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    * Returns the requires steps of this {@code AttackStep} object.
289    *
290    * @return the requires steps of this {@code AttackStep} object
291    * @since 1.0.0
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    * Returns whether this {@code AttackStep} object has local reaches steps.
311    *
312    * @return whether this {@code AttackStep} object has local reaches steps
313    * @since 1.0.0
314    */
315   public boolean hasLocalReaches() {
316     return this.reaches != null;
317   }
318 
319   /**
320    * Returns the local reaches steps of this {@code AttackStep} object.
321    *
322    * @return the local reaches steps of this {@code AttackStep} object
323    * @throws java.lang.UnsupportedOperationException if this {@code AttackStep} object does not have
324    *     local reaches steps
325    * @since 1.0.0
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    * Returns the reaches steps of this {@code AttackStep} object.
336    *
337    * @return the reaches steps of this {@code AttackStep} object
338    * @since 1.0.0
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    * Returns whether this {@code AttackStep} object has a super attack step.
358    *
359    * @return whether this {@code AttackStep} object has a super attack step
360    * @since 1.0.0
361    */
362   public boolean hasSuperAttackStep() {
363     return this.asset.hasSuperAsset() && this.asset.getSuperAsset().hasAttackStep(this.name);
364   }
365 
366   /**
367    * Returns the super attack step of this {@code AttackStep} object.
368    *
369    * @return the super attack step of this {@code AttackStep} object
370    * @throws java.lang.UnsupportedOperationException if this {@code AttackStep} object does not have
371    *     a super attack step
372    * @since 1.0.0
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 }