View Javadoc
1   /*
2    * Licensed to the Apache Software Foundation (ASF) under one or more
3    * contributor license agreements.  See the NOTICE file distributed with
4    * this work for additional information regarding copyright ownership.
5    * The ASF licenses this file to You under the Apache License, Version 2.0
6    * (the "License"); you may not use this file except in compliance with
7    * the License.  You may obtain a copy of the License at
8    *
9    *     https://www.apache.org/licenses/LICENSE-2.0
10   *
11   * Unless required by applicable law or agreed to in writing, software
12   * distributed under the License is distributed on an "AS IS" BASIS,
13   * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14   * See the License for the specific language governing permissions and
15   * limitations under the License.
16   */
17  package org.apache.commons.configuration2.tree;
18  
19  import java.util.Collection;
20  import java.util.List;
21  import java.util.concurrent.atomic.AtomicBoolean;
22  
23  /**
24   * <p>
25   * A specialized {@code NodeModel} implementation that uses a tracked node managed by an {@link InMemoryNodeModel}
26   * object as root node.
27   * </p>
28   * <p>
29   * Models of this type are useful when working on specific sub trees of a nodes structure. This is the case for instance
30   * for a {@code SubnodeConfiguration}.
31   * </p>
32   * <p>
33   * An instance of this class is constructed with an {@link InMemoryNodeModelSupport} object providing a reference to the
34   * underlying {@code InMemoryNodeModel} and the {@link NodeSelector} pointing to the tracked node acting as this model's
35   * root node. The {@code NodeModel} operations are implemented by delegating to the wrapped {@code InMemoryNodeModel}
36   * object specifying the selector to the tracked node as target root node for the update transaction. Note that the
37   * tracked node can become detached at any time. This situation is handled transparently by the implementation of
38   * {@code InMemoryNodeModel}. The reason for using an {@code InMemoryNodeModelSupport} object rather than an
39   * {@code InMemoryNodeModel} directly is that this additional layer of indirection can be used for performing special
40   * initializations on the model before it is returned to the {@code TrackedNodeModel} object. This is needed by some
41   * dynamic configuration implementations, for example by {@code CombinedConfiguration}.
42   * </p>
43   * <p>
44   * If the tracked node acting as root node is exclusively used by this model, it should be released when this model is
45   * no longer needed. This can be done manually by calling the {@link #close()} method. It is also possible to pass a
46   * value of <strong>true</strong> to the {@code untrackOnFinalize} argument of the constructor. This causes
47   * {@code close()} to be called automatically if this object gets claimed by the garbage collector.
48   * </p>
49   * <p>
50   * As {@code InMemoryNodeModel}, this class is thread-safe.
51   * </p>
52   *
53   * @since 2.0
54   */
55  public class TrackedNodeModel implements NodeModel<ImmutableNode> {
56  
57      /** Stores the underlying parent model. */
58      private final InMemoryNodeModelSupport parentModelSupport;
59  
60      /** The selector for the managed tracked node. */
61      private final NodeSelector selector;
62  
63      /**
64       * A flag whether the tracked not should be released when this object is finalized.
65       */
66      private final boolean releaseTrackedNodeOnFinalize;
67  
68      /** A flag whether this model has already been closed. */
69      private final AtomicBoolean closed;
70  
71      /**
72       * Creates a new instance of {@code TrackedNodeModel} and initializes it with the given underlying model and the
73       * selector to the root node. The boolean argument controls whether the associated tracked node should be released when
74       * this object gets finalized. This allows the underlying model to free some resources. If used as model within a
75       * {@code SubnodeConfiguration}, there is typically no way to discard the model explicitly. Therefore, it makes sense to
76       * do this automatically on finalization.
77       *
78       * @param modelSupport the underlying {@code InMemoryNodeModelSupport} (must not be <strong>null</strong>)
79       * @param sel the selector to the root node of this model (must not be <strong>null</strong>)
80       * @param untrackOnFinalize a flag whether the tracked node should be released on finalization
81       * @throws IllegalArgumentException if a required parameter is missing
82       */
83      public TrackedNodeModel(final InMemoryNodeModelSupport modelSupport, final NodeSelector sel, final boolean untrackOnFinalize) {
84          if (modelSupport == null) {
85              throw new IllegalArgumentException("Underlying model support must not be null!");
86          }
87          if (sel == null) {
88              throw new IllegalArgumentException("Selector must not be null!");
89          }
90  
91          parentModelSupport = modelSupport;
92          selector = sel;
93          releaseTrackedNodeOnFinalize = untrackOnFinalize;
94          closed = new AtomicBoolean();
95      }
96  
97      @Override
98      public void addNodes(final String key, final Collection<? extends ImmutableNode> nodes, final NodeKeyResolver<ImmutableNode> resolver) {
99          getParentModel().addNodes(key, getSelector(), nodes, resolver);
100     }
101 
102     @Override
103     public void addProperty(final String key, final Iterable<?> values, final NodeKeyResolver<ImmutableNode> resolver) {
104         getParentModel().addProperty(key, getSelector(), values, resolver);
105     }
106 
107     /**
108      * {@inheritDoc} This implementation clears the sub tree spanned by the associate tracked node. This has the side effect
109      * that this in any case becomes detached.
110      *
111      * @param resolver the {@code NodeKeyResolver}.
112      */
113     @Override
114     public void clear(final NodeKeyResolver<ImmutableNode> resolver) {
115         getParentModel().clearTree(null, getSelector(), resolver);
116     }
117 
118     @Override
119     public void clearProperty(final String key, final NodeKeyResolver<ImmutableNode> resolver) {
120         getParentModel().clearProperty(key, getSelector(), resolver);
121     }
122 
123     @Override
124     public List<QueryResult<ImmutableNode>> clearTree(final String key, final NodeKeyResolver<ImmutableNode> resolver) {
125         return getParentModel().clearTree(key, getSelector(), resolver);
126     }
127 
128     /**
129      * Closes this model. This causes the tracked node this model is based upon to be released (i.e.
130      * {@link InMemoryNodeModel#untrackNode(NodeSelector)} is called). This method should be called when this model is no
131      * longer needed. This implementation is idempotent; it is safe to call {@code close()} multiple times - only the first
132      * invocation has an effect. After this method has been called this model can no longer be used because there is no
133      * guarantee that the node can still be accessed from the parent model.
134      */
135     public void close() {
136         if (closed.compareAndSet(false, true)) {
137             getParentModel().untrackNode(getSelector());
138         }
139     }
140 
141     /**
142      * {@inheritDoc} This implementation calls {@code close()} if the {@code untrackOnFinalize} flag was set when this
143      * instance was constructed. While this is not 100 percent reliable, it is better than keeping the tracked node hanging
144      * around. Note that it is not a problem if {@code close()} already had been invoked manually because this method is
145      * idempotent.
146      *
147      * @see #close()
148      */
149     @Override
150     protected void finalize() throws Throwable {
151         if (isReleaseTrackedNodeOnFinalize()) {
152             close();
153         }
154         super.finalize();
155     }
156 
157     /**
158      * {@inheritDoc} This implementation returns the tracked node instance acting as root node of this model.
159      */
160     @Override
161     public ImmutableNode getInMemoryRepresentation() {
162         return getNodeHandler().getRootNode();
163     }
164 
165     @Override
166     public NodeHandler<ImmutableNode> getNodeHandler() {
167         return getParentModel().getTrackedNodeHandler(getSelector());
168     }
169 
170     /**
171      * Gets the parent model. Operations on this model are delegated to this parent model specifying the selector to the
172      * tracked node.
173      *
174      * @return the parent model
175      */
176     public InMemoryNodeModel getParentModel() {
177         return getParentModelSupport().getNodeModel();
178     }
179 
180     /**
181      * Gets the {@code InMemoryNodeModelSupport} object which is used to gain access to the underlying node model.
182      *
183      * @return the associated {@code InMemoryNodeModelSupport} object
184      */
185     public InMemoryNodeModelSupport getParentModelSupport() {
186         return parentModelSupport;
187     }
188 
189     /**
190      * Gets the {@code NodeSelector} pointing to the tracked node managed by this model.
191      *
192      * @return the tracked node selector
193      */
194     public NodeSelector getSelector() {
195         return selector;
196     }
197 
198     /**
199      * Returns the flag whether the managed tracked node is to be released when this object gets finalized. This method
200      * returns the value of the corresponding flag passed to the constructor. If result is true, the underlying model is
201      * asked to untrack the managed node when this object is claimed by the GC.
202      *
203      * @return a flag whether the managed tracked node should be released when this object dies
204      * @see InMemoryNodeModel#untrackNode(NodeSelector)
205      */
206     public boolean isReleaseTrackedNodeOnFinalize() {
207         return releaseTrackedNodeOnFinalize;
208     }
209 
210     @Override
211     public void setProperty(final String key, final Object value, final NodeKeyResolver<ImmutableNode> resolver) {
212         getParentModel().setProperty(key, getSelector(), value, resolver);
213     }
214 
215     @Override
216     public void setRootNode(final ImmutableNode newRoot) {
217         getParentModel().replaceTrackedNode(getSelector(), newRoot);
218     }
219 }