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.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
32
33
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
60
61
62
63
64
65
66 public boolean hasDefine(String key) {
67 return this.defines.containsKey(requireNonNull(key));
68 }
69
70
71
72
73
74
75
76
77
78
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
89
90
91
92
93 public Map<String, String> getDefines() {
94 return this.defines;
95 }
96
97
98
99
100
101
102
103
104
105 public boolean hasCategory(String name) {
106 return this.categories.containsKey(requireNonNull(name));
107 }
108
109
110
111
112
113
114
115
116
117
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
128
129
130
131
132 public List<Category> getCategories() {
133 return List.copyOf(this.categories.values());
134 }
135
136
137
138
139
140
141
142
143
144 public boolean hasAsset(String name) {
145 return this.assets.containsKey(requireNonNull(name));
146 }
147
148
149
150
151
152
153
154
155
156
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
167
168
169
170
171 public List<Asset> getAssets() {
172 return List.copyOf(this.assets.values());
173 }
174
175
176
177
178
179
180
181 public List<Association> getAssociations() {
182 return this.associations;
183 }
184
185
186
187
188
189
190
191 public boolean hasLicense() {
192 return this.license != null;
193 }
194
195
196
197
198
199
200
201
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
212
213
214
215
216 public boolean hasNotice() {
217 return this.notice != null;
218 }
219
220
221
222
223
224
225
226
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
237
238
239
240
241 public JsonObject toJson() {
242
243 var jsonDefines = Json.createObjectBuilder();
244 for (var define : this.defines.entrySet()) {
245 jsonDefines.add(define.getKey(), define.getValue());
246 }
247
248
249 var jsonCategories = Json.createArrayBuilder();
250 for (var category : this.categories.values()) {
251 jsonCategories.add(category.toJson());
252 }
253
254
255 var jsonAssets = Json.createArrayBuilder();
256 for (var asset : this.assets.values()) {
257 jsonAssets.add(asset.toJson());
258 }
259
260
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
277
278
279
280
281
282
283 public static Lang fromBuilder(LangBuilder builder) {
284 requireNonNull(builder);
285
286
287 Map<String, Category> categories = new LinkedHashMap<>();
288 for (var categoryBuilder : builder.getCategories()) {
289 categories.put(categoryBuilder.getName(), Category.fromBuilder(categoryBuilder));
290 }
291
292
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
309 List<Association> associations = new ArrayList<>();
310 for (var associationBuilder : builder.getAssociations()) {
311 associations.add(Association.fromBuilder(associationBuilder, assets));
312 }
313
314
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
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 }