1 /*
2  * Copyright 2017, OpenCensus Authors
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 io.opencensus.stats;
18 
19 import io.opencensus.common.Functions;
20 import io.opencensus.common.Timestamp;
21 import io.opencensus.internal.Utils;
22 import io.opencensus.stats.Measure.MeasureDouble;
23 import io.opencensus.stats.Measure.MeasureLong;
24 import io.opencensus.tags.TagContext;
25 import io.opencensus.tags.TagValue;
26 import java.util.Collection;
27 import java.util.Collections;
28 import java.util.HashMap;
29 import java.util.HashSet;
30 import java.util.List;
31 import java.util.Map;
32 import java.util.Set;
33 import javax.annotation.concurrent.GuardedBy;
34 import javax.annotation.concurrent.Immutable;
35 import javax.annotation.concurrent.ThreadSafe;
36 
37 /*>>>
38 import org.checkerframework.checker.nullness.qual.Nullable;
39 */
40 
41 /** No-op implementations of stats classes. */
42 final class NoopStats {
43 
NoopStats()44   private NoopStats() {}
45 
46   /**
47    * Returns a {@code StatsComponent} that has a no-op implementation for {@link StatsRecorder}.
48    *
49    * @return a {@code StatsComponent} that has a no-op implementation for {@code StatsRecorder}.
50    */
newNoopStatsComponent()51   static StatsComponent newNoopStatsComponent() {
52     return new NoopStatsComponent();
53   }
54 
55   /**
56    * Returns a {@code StatsRecorder} that does not record any data.
57    *
58    * @return a {@code StatsRecorder} that does not record any data.
59    */
getNoopStatsRecorder()60   static StatsRecorder getNoopStatsRecorder() {
61     return NoopStatsRecorder.INSTANCE;
62   }
63 
64   /**
65    * Returns a {@code MeasureMap} that ignores all calls to {@link MeasureMap#put}.
66    *
67    * @return a {@code MeasureMap} that ignores all calls to {@code MeasureMap#put}.
68    */
getNoopMeasureMap()69   static MeasureMap getNoopMeasureMap() {
70     return NoopMeasureMap.INSTANCE;
71   }
72 
73   /**
74    * Returns a {@code ViewManager} that maintains a map of views, but always returns empty {@link
75    * ViewData}s.
76    *
77    * @return a {@code ViewManager} that maintains a map of views, but always returns empty {@code
78    *     ViewData}s.
79    */
newNoopViewManager()80   static ViewManager newNoopViewManager() {
81     return new NoopViewManager();
82   }
83 
84   @ThreadSafe
85   private static final class NoopStatsComponent extends StatsComponent {
86     private final ViewManager viewManager = newNoopViewManager();
87     private volatile boolean isRead;
88 
89     @Override
getViewManager()90     public ViewManager getViewManager() {
91       return viewManager;
92     }
93 
94     @Override
getStatsRecorder()95     public StatsRecorder getStatsRecorder() {
96       return getNoopStatsRecorder();
97     }
98 
99     @Override
getState()100     public StatsCollectionState getState() {
101       isRead = true;
102       return StatsCollectionState.DISABLED;
103     }
104 
105     @Override
106     @Deprecated
setState(StatsCollectionState state)107     public void setState(StatsCollectionState state) {
108       Utils.checkNotNull(state, "state");
109       Utils.checkState(!isRead, "State was already read, cannot set state.");
110     }
111   }
112 
113   @Immutable
114   private static final class NoopStatsRecorder extends StatsRecorder {
115     static final StatsRecorder INSTANCE = new NoopStatsRecorder();
116 
117     @Override
newMeasureMap()118     public MeasureMap newMeasureMap() {
119       return getNoopMeasureMap();
120     }
121   }
122 
123   @Immutable
124   private static final class NoopMeasureMap extends MeasureMap {
125     static final MeasureMap INSTANCE = new NoopMeasureMap();
126 
127     @Override
put(MeasureDouble measure, double value)128     public MeasureMap put(MeasureDouble measure, double value) {
129       return this;
130     }
131 
132     @Override
put(MeasureLong measure, long value)133     public MeasureMap put(MeasureLong measure, long value) {
134       return this;
135     }
136 
137     @Override
record()138     public void record() {}
139 
140     @Override
record(TagContext tags)141     public void record(TagContext tags) {
142       Utils.checkNotNull(tags, "tags");
143     }
144   }
145 
146   @ThreadSafe
147   private static final class NoopViewManager extends ViewManager {
148     private static final Timestamp ZERO_TIMESTAMP = Timestamp.create(0, 0);
149 
150     @GuardedBy("registeredViews")
151     private final Map<View.Name, View> registeredViews = new HashMap<View.Name, View>();
152 
153     // Cached set of exported views. It must be set to null whenever a view is registered or
154     // unregistered.
155     @javax.annotation.Nullable private volatile Set<View> exportedViews;
156 
157     @Override
registerView(View newView)158     public void registerView(View newView) {
159       Utils.checkNotNull(newView, "newView");
160       synchronized (registeredViews) {
161         exportedViews = null;
162         View existing = registeredViews.get(newView.getName());
163         Utils.checkArgument(
164             existing == null || newView.equals(existing),
165             "A different view with the same name already exists.");
166         if (existing == null) {
167           registeredViews.put(newView.getName(), newView);
168         }
169       }
170     }
171 
172     @Override
173     @javax.annotation.Nullable
174     @SuppressWarnings("deprecation")
getView(View.Name name)175     public ViewData getView(View.Name name) {
176       Utils.checkNotNull(name, "name");
177       synchronized (registeredViews) {
178         View view = registeredViews.get(name);
179         if (view == null) {
180           return null;
181         } else {
182           return ViewData.create(
183               view,
184               Collections.<List</*@Nullable*/ TagValue>, AggregationData>emptyMap(),
185               view.getWindow()
186                   .match(
187                       Functions.<ViewData.AggregationWindowData>returnConstant(
188                           ViewData.AggregationWindowData.CumulativeData.create(
189                               ZERO_TIMESTAMP, ZERO_TIMESTAMP)),
190                       Functions.<ViewData.AggregationWindowData>returnConstant(
191                           ViewData.AggregationWindowData.IntervalData.create(ZERO_TIMESTAMP)),
192                       Functions.<ViewData.AggregationWindowData>throwAssertionError()));
193         }
194       }
195     }
196 
197     @Override
getAllExportedViews()198     public Set<View> getAllExportedViews() {
199       Set<View> views = exportedViews;
200       if (views == null) {
201         synchronized (registeredViews) {
202           exportedViews = views = filterExportedViews(registeredViews.values());
203         }
204       }
205       return views;
206     }
207 
208     // Returns the subset of the given views that should be exported
209     @SuppressWarnings("deprecation")
filterExportedViews(Collection<View> allViews)210     private static Set<View> filterExportedViews(Collection<View> allViews) {
211       Set<View> views = new HashSet<View>();
212       for (View view : allViews) {
213         if (view.getWindow() instanceof View.AggregationWindow.Interval) {
214           continue;
215         }
216         views.add(view);
217       }
218       return Collections.unmodifiableSet(views);
219     }
220   }
221 }
222