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.LinkedHashMap;
25  import java.util.List;
26  import java.util.Map;
27  import org.mal_lang.langspec.builders.LangBuilder;
28  import org.mal_lang.langspec.step.StepExpression;
29  
30  /**
31   * Immutable class representing a MAL language.
32   *
33   * @since 1.0.0
34   */
35  public final class Lang {
36    private final Map<String, String> defines;
37    private final Map<String, Category> categories;
38    private final Map<String, Asset> assets;
39    private final List<Association> associations;
40    private final String license;
41    private final String notice;
42  
43    private Lang(
44        Map<String, String> defines,
45        Map<String, Category> categories,
46        Map<String, Asset> assets,
47        List<Association> associations,
48        String license,
49        String notice) {
50      this.defines = Map.copyOf(requireNonNull(defines));
51      this.categories = new LinkedHashMap<>(requireNonNull(categories));
52      this.assets = new LinkedHashMap<>(requireNonNull(assets));
53      this.associations = List.copyOf(requireNonNull(associations));
54      this.license = license;
55      this.notice = notice;
56    }
57  
58    /**
59     * Returns whether {@code key} is the key of a define in this {@code Lang} object.
60     *
61     * @param key the key of the define
62     * @return whether {@code key} is the key of a define in this {@code Lang} object
63     * @throws java.lang.NullPointerException if {@code key} is {@code null}
64     * @since 1.0.0
65     */
66    public boolean hasDefine(String key) {
67      return this.defines.containsKey(requireNonNull(key));
68    }
69  
70    /**
71     * Returns the value of the define with the key {@code key} in this {@code Lang} object.
72     *
73     * @param key the key of the define
74     * @return the value of the define with the key {@code key} in this {@code Lang} object
75     * @throws java.lang.NullPointerException if {@code key} is {@code null}
76     * @throws java.lang.IllegalArgumentException if {@code key} is not the key of a define in this
77     *     {@code Lang} object
78     * @since 1.0.0
79     */
80    public String getDefine(String key) {
81      if (!this.hasDefine(key)) {
82        throw new IllegalArgumentException(String.format("Define \"%s\" not found", key));
83      }
84      return this.defines.get(key);
85    }
86  
87    /**
88     * Returns all defines in this {@code Lang} object.
89     *
90     * @return all defines in this {@code Lang} object
91     * @since 1.0.0
92     */
93    public Map<String, String> getDefines() {
94      return this.defines;
95    }
96  
97    /**
98     * Returns whether {@code name} is the name of a category in this {@code Lang} object.
99     *
100    * @param name the name of the category
101    * @return whether {@code name} is the name of a category in this {@code Lang} object
102    * @throws java.lang.NullPointerException if {@code name} is {@code null}
103    * @since 1.0.0
104    */
105   public boolean hasCategory(String name) {
106     return this.categories.containsKey(requireNonNull(name));
107   }
108 
109   /**
110    * Returns the category with the name {@code name} in this {@code Lang} object.
111    *
112    * @param name the name of the category
113    * @return the category with the name {@code name} in this {@code Lang} object
114    * @throws java.lang.NullPointerException if {@code name} is {@code null}
115    * @throws java.lang.IllegalArgumentException if {@code name} is not the name of a category in
116    *     this {@code Lang} object
117    * @since 1.0.0
118    */
119   public Category getCategory(String name) {
120     if (!this.hasCategory(name)) {
121       throw new IllegalArgumentException(String.format("Category \"%s\" not found", name));
122     }
123     return this.categories.get(name);
124   }
125 
126   /**
127    * Returns a list of all categories in this {@code Lang} object.
128    *
129    * @return a list of all categories in this {@code Lang} object
130    * @since 1.0.0
131    */
132   public List<Category> getCategories() {
133     return List.copyOf(this.categories.values());
134   }
135 
136   /**
137    * Returns whether {@code name} is the name of an asset in this {@code Lang} object.
138    *
139    * @param name the name of the asset
140    * @return whether {@code name} is the name of an asset in this {@code Lang} object
141    * @throws java.lang.NullPointerException if {@code name} is {@code null}
142    * @since 1.0.0
143    */
144   public boolean hasAsset(String name) {
145     return this.assets.containsKey(requireNonNull(name));
146   }
147 
148   /**
149    * Returns the asset with the name {@code name} in this {@code Lang} object.
150    *
151    * @param name the name of the asset
152    * @return the asset with the name {@code name} in this {@code Lang} object
153    * @throws java.lang.NullPointerException if {@code name} is {@code null}
154    * @throws java.lang.IllegalArgumentException if {@code name} is not the name of an asset in this
155    *     {@code Lang} object
156    * @since 1.0.0
157    */
158   public Asset getAsset(String name) {
159     if (!this.hasAsset(name)) {
160       throw new IllegalArgumentException(String.format("Asset \"%s\" not found", name));
161     }
162     return this.assets.get(name);
163   }
164 
165   /**
166    * Returns a list of all assets in this {@code Lang} object.
167    *
168    * @return a list of all assets in this {@code Lang} object
169    * @since 1.0.0
170    */
171   public List<Asset> getAssets() {
172     return List.copyOf(this.assets.values());
173   }
174 
175   /**
176    * Returns a list of all associations in this {@code Lang} object.
177    *
178    * @return a list of all associations in this {@code Lang} object
179    * @since 1.0.0
180    */
181   public List<Association> getAssociations() {
182     return this.associations;
183   }
184 
185   /**
186    * Returns whether this {@code Lang} object has a license.
187    *
188    * @return whether this {@code Lang} object has a license
189    * @since 1.0.0
190    */
191   public boolean hasLicense() {
192     return this.license != null;
193   }
194 
195   /**
196    * Returns the license of this {@code Lang} object.
197    *
198    * @return the license of this {@code Lang} object
199    * @throws java.lang.UnsupportedOperationException if this {@code Lang} object does not have a
200    *     license
201    * @since 1.0.0
202    */
203   public String getLicense() {
204     if (!this.hasLicense()) {
205       throw new UnsupportedOperationException("License not found");
206     }
207     return this.license;
208   }
209 
210   /**
211    * Returns whether this {@code Lang} object has a notice.
212    *
213    * @return whether this {@code Lang} object has a notice
214    * @since 1.0.0
215    */
216   public boolean hasNotice() {
217     return this.notice != null;
218   }
219 
220   /**
221    * Returns the notice of this {@code Lang} object.
222    *
223    * @return the notice of this {@code Lang} object
224    * @throws java.lang.UnsupportedOperationException if this {@code Lang} object does not have a
225    *     notice
226    * @since 1.0.0
227    */
228   public String getNotice() {
229     if (!this.hasNotice()) {
230       throw new UnsupportedOperationException("Notice not found");
231     }
232     return this.notice;
233   }
234 
235   /**
236    * Returns the JSON representation of this {@code Lang} object.
237    *
238    * @return the JSON representation of this {@code Lang} object
239    * @since 1.0.0
240    */
241   public JsonObject toJson() {
242     // Defines
243     var jsonDefines = Json.createObjectBuilder();
244     for (var define : this.defines.entrySet()) {
245       jsonDefines.add(define.getKey(), define.getValue());
246     }
247 
248     // Categories
249     var jsonCategories = Json.createArrayBuilder();
250     for (var category : this.categories.values()) {
251       jsonCategories.add(category.toJson());
252     }
253 
254     // Assets
255     var jsonAssets = Json.createArrayBuilder();
256     for (var asset : this.assets.values()) {
257       jsonAssets.add(asset.toJson());
258     }
259 
260     // Associations
261     var jsonAssociations = Json.createArrayBuilder();
262     for (var association : this.associations) {
263       jsonAssociations.add(association.toJson());
264     }
265 
266     return Json.createObjectBuilder()
267         .add("formatVersion", Utils.getFormatVersion())
268         .add("defines", jsonDefines)
269         .add("categories", jsonCategories)
270         .add("assets", jsonAssets)
271         .add("associations", jsonAssociations)
272         .build();
273   }
274 
275   /**
276    * Creates a new {@code Lang} object from a {@link org.mal_lang.langspec.builders.LangBuilder}.
277    *
278    * @param builder the {@link org.mal_lang.langspec.builders.LangBuilder}
279    * @return a new {@code Lang} object
280    * @throws java.lang.NullPointerException if {@code builder} is {@code null}
281    * @since 1.0.0
282    */
283   public static Lang fromBuilder(LangBuilder builder) {
284     requireNonNull(builder);
285 
286     // Categories
287     Map<String, Category> categories = new LinkedHashMap<>();
288     for (var categoryBuilder : builder.getCategories()) {
289       categories.put(categoryBuilder.getName(), Category.fromBuilder(categoryBuilder));
290     }
291 
292     // Assets
293     Map<String, Asset> assets = new LinkedHashMap<>();
294     for (var assetBuilder : builder.getAssets()) {
295       assets.put(assetBuilder.getName(), Asset.fromBuilder(assetBuilder, categories));
296     }
297     for (var assetBuilder : builder.getAssets()) {
298       if (assetBuilder.getSuperAsset() == null) {
299         continue;
300       }
301       if (!assets.containsKey(assetBuilder.getSuperAsset())) {
302         throw new IllegalArgumentException(
303             String.format("Asset \"%s\" not found", assetBuilder.getSuperAsset()));
304       }
305       assets.get(assetBuilder.getName()).setSuperAsset(assets.get(assetBuilder.getSuperAsset()));
306     }
307 
308     // Associations
309     List<Association> associations = new ArrayList<>();
310     for (var associationBuilder : builder.getAssociations()) {
311       associations.add(Association.fromBuilder(associationBuilder, assets));
312     }
313 
314     // Variable targets
315     var variableTargets = new LinkedHashMap<Variable, Asset>();
316     for (var assetBuilder : builder.getAssets()) {
317       var asset = assets.get(assetBuilder.getName());
318       for (var variableBuilder : assetBuilder.getVariables()) {
319         var variable = asset.getLocalVariable(variableBuilder.getName());
320         var targetAsset =
321             variableBuilder.getStepExpression().getTarget(asset, assets, builder.getAssets());
322         variableTargets.put(variable, targetAsset);
323       }
324     }
325 
326     // Variables and attack steps
327     for (var assetBuilder : builder.getAssets()) {
328       var asset = assets.get(assetBuilder.getName());
329       for (var variableBuilder : assetBuilder.getVariables()) {
330         var variable = asset.getLocalVariable(variableBuilder.getName());
331         variable.setStepExpression(
332             StepExpression.fromBuilder(
333                 variableBuilder.getStepExpression(), asset, assets, variableTargets));
334       }
335       for (var attackStepBuilder : assetBuilder.getAttackSteps()) {
336         var attackStep = asset.getLocalAttackStep(attackStepBuilder.getName());
337         if (attackStepBuilder.getRequires() != null) {
338           for (var stepExpressionBuilder : attackStepBuilder.getRequires().getStepExpressions()) {
339             attackStep
340                 .getLocalRequires()
341                 .addStepExpression(
342                     StepExpression.fromBuilder(
343                         stepExpressionBuilder, asset, assets, variableTargets));
344           }
345         }
346         if (attackStepBuilder.getReaches() != null) {
347           for (var stepExpressionBuilder : attackStepBuilder.getReaches().getStepExpressions()) {
348             attackStep
349                 .getLocalReaches()
350                 .addStepExpression(
351                     StepExpression.fromBuilder(
352                         stepExpressionBuilder, asset, assets, variableTargets));
353           }
354         }
355       }
356     }
357 
358     return new Lang(
359         builder.getDefines(),
360         categories,
361         assets,
362         associations,
363         builder.getLicense(),
364         builder.getNotice());
365   }
366 }