View Javadoc
1   /*
2    * Portions of this software was developed by employees of the National Institute
3    * of Standards and Technology (NIST), an agency of the Federal Government and is
4    * being made available as a public service. Pursuant to title 17 United States
5    * Code Section 105, works of NIST employees are not subject to copyright
6    * protection in the United States. This software may be subject to foreign
7    * copyright. Permission in the United States and in foreign countries, to the
8    * extent that NIST may hold copyright, to use, copy, modify, create derivative
9    * works, and distribute this software and its documentation without fee is hereby
10   * granted on a non-exclusive basis, provided that this notice and disclaimer
11   * of warranty appears in all copies.
12   *
13   * THE SOFTWARE IS PROVIDED 'AS IS' WITHOUT ANY WARRANTY OF ANY KIND, EITHER
14   * EXPRESSED, IMPLIED, OR STATUTORY, INCLUDING, BUT NOT LIMITED TO, ANY WARRANTY
15   * THAT THE SOFTWARE WILL CONFORM TO SPECIFICATIONS, ANY IMPLIED WARRANTIES OF
16   * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE, AND FREEDOM FROM
17   * INFRINGEMENT, AND ANY WARRANTY THAT THE DOCUMENTATION WILL CONFORM TO THE
18   * SOFTWARE, OR ANY WARRANTY THAT THE SOFTWARE WILL BE ERROR FREE.  IN NO EVENT
19   * SHALL NIST BE LIABLE FOR ANY DAMAGES, INCLUDING, BUT NOT LIMITED TO, DIRECT,
20   * INDIRECT, SPECIAL OR CONSEQUENTIAL DAMAGES, ARISING OUT OF, RESULTING FROM,
21   * OR IN ANY WAY CONNECTED WITH THIS SOFTWARE, WHETHER OR NOT BASED UPON WARRANTY,
22   * CONTRACT, TORT, OR OTHERWISE, WHETHER OR NOT INJURY WAS SUSTAINED BY PERSONS OR
23   * PROPERTY OR OTHERWISE, AND WHETHER OR NOT LOSS WAS SUSTAINED FROM, OR AROSE OUT
24   * OF THE RESULTS OF, OR USE OF, THE SOFTWARE OR SERVICES PROVIDED HEREUNDER.
25   */
26  
27  package gov.nist.secauto.metaschema.core.model.xml;
28  
29  import gov.nist.secauto.metaschema.core.model.IAssemblyDefinition;
30  import gov.nist.secauto.metaschema.core.model.IFieldDefinition;
31  import gov.nist.secauto.metaschema.core.model.IFlagContainer;
32  import gov.nist.secauto.metaschema.core.model.IFlagContainerSupport;
33  import gov.nist.secauto.metaschema.core.model.IFlagInstance;
34  import gov.nist.secauto.metaschema.core.model.xml.xmlbeans.FlagReferenceType;
35  import gov.nist.secauto.metaschema.core.model.xml.xmlbeans.GlobalAssemblyDefinitionType;
36  import gov.nist.secauto.metaschema.core.model.xml.xmlbeans.GlobalFieldDefinitionType;
37  import gov.nist.secauto.metaschema.core.model.xml.xmlbeans.InlineAssemblyDefinitionType;
38  import gov.nist.secauto.metaschema.core.model.xml.xmlbeans.InlineFieldDefinitionType;
39  import gov.nist.secauto.metaschema.core.model.xml.xmlbeans.InlineFlagDefinitionType;
40  import gov.nist.secauto.metaschema.core.util.CollectionUtil;
41  
42  import org.apache.xmlbeans.XmlCursor;
43  import org.apache.xmlbeans.XmlObject;
44  
45  import java.util.Collections;
46  import java.util.LinkedHashMap;
47  import java.util.Map;
48  
49  import edu.umd.cs.findbugs.annotations.NonNull;
50  import edu.umd.cs.findbugs.annotations.Nullable;
51  
52  @SuppressWarnings("PMD.NullAssignment") // readability
53  class XmlFlagContainerSupport implements IFlagContainerSupport<IFlagInstance> {
54  
55    @NonNull
56    private final Map<String, IFlagInstance> flagInstances;
57    @Nullable
58    private final IFlagInstance jsonKeyFlag;
59  
60    /**
61     * Generate a set of constraints from the provided XMLBeans instance.
62     *
63     * @param xmlField
64     *          the XMLBeans instance
65     * @param container
66     *          the field containing the flag
67     */
68    public XmlFlagContainerSupport(
69        @NonNull GlobalFieldDefinitionType xmlField,
70        @NonNull IFieldDefinition container) {
71      // handle flags
72      if (xmlField.getFlagList().size() > 0 || xmlField.getDefineFlagList().size() > 0) {
73        this.flagInstances = parseLocalFlags(xmlField, container);
74      } else {
75        this.flagInstances = CollectionUtil.emptyMap();
76      }
77      this.jsonKeyFlag = xmlField.isSetJsonKey() ? flagInstances.get(xmlField.getJsonKey().getFlagRef()) : null;
78    }
79  
80    /**
81     * Generate a set of constraints from the provided XMLBeans instance.
82     *
83     * @param xmlField
84     *          the XMLBeans instance
85     * @param container
86     *          the field containing the flag
87     */
88    public XmlFlagContainerSupport(
89        @NonNull InlineFieldDefinitionType xmlField,
90        @NonNull IFieldDefinition container) {
91      // handle flags
92      if (xmlField.getFlagList().size() > 0 || xmlField.getDefineFlagList().size() > 0) {
93        this.flagInstances = parseLocalFlags(xmlField, container);
94      } else {
95        this.flagInstances = CollectionUtil.emptyMap();
96      }
97      this.jsonKeyFlag = xmlField.isSetJsonKey() ? flagInstances.get(xmlField.getJsonKey().getFlagRef()) : null;
98    }
99  
100   /**
101    * Generate a set of constraints from the provided XMLBeans instance.
102    *
103    * @param xmlAssembly
104    *          the XMLBeans instance
105    * @param container
106    *          the assembly containing the flag
107    */
108   public XmlFlagContainerSupport(
109       @NonNull GlobalAssemblyDefinitionType xmlAssembly,
110       @NonNull IAssemblyDefinition container) {
111     // handle flags
112     if (xmlAssembly.getFlagList().size() > 0 || xmlAssembly.getDefineFlagList().size() > 0) {
113       this.flagInstances = parseLocalFlags(xmlAssembly, container);
114     } else {
115       this.flagInstances = CollectionUtil.emptyMap();
116     }
117     this.jsonKeyFlag = xmlAssembly.isSetJsonKey() ? flagInstances.get(xmlAssembly.getJsonKey().getFlagRef()) : null;
118   }
119 
120   /**
121    * Generate a set of constraints from the provided XMLBeans instance.
122    *
123    * @param xmlAssembly
124    *          the XMLBeans instance
125    * @param container
126    *          the assembly containing the flag
127    */
128   public XmlFlagContainerSupport(
129       @NonNull InlineAssemblyDefinitionType xmlAssembly,
130       @NonNull IAssemblyDefinition container) {
131     // handle flags
132     if (xmlAssembly.getFlagList().size() > 0 || xmlAssembly.getDefineFlagList().size() > 0) {
133       this.flagInstances = parseLocalFlags(xmlAssembly, container);
134     } else {
135       this.flagInstances = CollectionUtil.emptyMap();
136     }
137     this.jsonKeyFlag = xmlAssembly.isSetJsonKey() ? flagInstances.get(xmlAssembly.getJsonKey().getFlagRef()) : null;
138   }
139 
140   /**
141    * Get a mapping of flag effective name to flag instance.
142    *
143    * @return the mapping of flag effective name to flag instance
144    */
145   @Override
146   @NonNull
147   public Map<String, ? extends IFlagInstance> getFlagInstanceMap() {
148     return flagInstances;
149   }
150 
151   @Override
152   public IFlagInstance getJsonKeyFlagInstance() {
153     return jsonKeyFlag;
154   }
155 
156   @NonNull
157   private static Map<String, IFlagInstance> parseLocalFlags(@NonNull XmlObject xmlObject,
158       @NonNull IFlagContainer parent) {
159     // handle flags
160     Map<String, IFlagInstance> flagInstances = new LinkedHashMap<>(); // NOPMD - intentional
161     try (XmlCursor cursor = xmlObject.newCursor()) {
162       cursor.selectPath(
163           "declare namespace m='http://csrc.nist.gov/ns/oscal/metaschema/1.0';" + "$this/m:flag|$this/m:define-flag");
164 
165       while (cursor.toNextSelection()) {
166         XmlObject obj = cursor.getObject();
167         if (obj instanceof FlagReferenceType) {
168           XmlFlagInstance flagInstance = new XmlFlagInstance((FlagReferenceType) obj, parent); // NOPMD - intentional
169           flagInstances.put(flagInstance.getEffectiveName(), flagInstance);
170         } else if (obj instanceof InlineFlagDefinitionType) {
171           XmlInlineFlagDefinition flagInstance
172               = new XmlInlineFlagDefinition((InlineFlagDefinitionType) obj, parent); // NOPMD - intentional
173           flagInstances.put(flagInstance.getEffectiveName(), flagInstance);
174         }
175       }
176     }
177 
178     @SuppressWarnings("null")
179     @NonNull Map<String, IFlagInstance> retval
180         = flagInstances.isEmpty() ? Collections.emptyMap() : Collections.unmodifiableMap(flagInstances);
181     return retval;
182   }
183 }