16
16
17
17
package com .google .cloud .spanner ;
18
18
19
+ import static com .google .cloud .spanner .MetricRegistryConstants .COUNT ;
20
+ import static com .google .cloud .spanner .MetricRegistryConstants .IN_USE_SESSIONS ;
21
+ import static com .google .cloud .spanner .MetricRegistryConstants .IN_USE_SESSIONS_DESCRIPTION ;
22
+ import static com .google .cloud .spanner .MetricRegistryConstants .MAX_ALLOWED_SESSIONS ;
23
+ import static com .google .cloud .spanner .MetricRegistryConstants .MAX_ALLOWED_SESSIONS_DESCRIPTION ;
24
+ import static com .google .cloud .spanner .MetricRegistryConstants .MAX_IN_USE_SESSIONS ;
25
+ import static com .google .cloud .spanner .MetricRegistryConstants .MAX_IN_USE_SESSIONS_DESCRIPTION ;
26
+ import static com .google .cloud .spanner .MetricRegistryConstants .SPANNER_DEFAULT_LABEL_VALUES ;
27
+ import static com .google .cloud .spanner .MetricRegistryConstants .SPANNER_LABEL_KEYS ;
19
28
import static com .google .cloud .spanner .SpannerExceptionFactory .newSpannerException ;
20
29
21
30
import com .google .api .core .ApiFuture ;
40
49
import com .google .common .util .concurrent .Uninterruptibles ;
41
50
import com .google .protobuf .Empty ;
42
51
import io .opencensus .common .Scope ;
52
+ import io .opencensus .common .ToLongFunction ;
53
+ import io .opencensus .metrics .DerivedLongGauge ;
54
+ import io .opencensus .metrics .LabelValue ;
55
+ import io .opencensus .metrics .MetricOptions ;
56
+ import io .opencensus .metrics .MetricRegistry ;
57
+ import io .opencensus .metrics .Metrics ;
43
58
import io .opencensus .trace .Annotation ;
44
59
import io .opencensus .trace .AttributeValue ;
45
60
import io .opencensus .trace .Span ;
49
64
import java .util .HashSet ;
50
65
import java .util .Iterator ;
51
66
import java .util .LinkedList ;
67
+ import java .util .List ;
52
68
import java .util .Queue ;
53
69
import java .util .Random ;
54
70
import java .util .Set ;
@@ -1116,11 +1132,15 @@ private static enum Position {
1116
1132
* Return pool is immediately ready for use, though getting a session might block for sessions to
1117
1133
* be created.
1118
1134
*/
1119
- static SessionPool createPool (SpannerOptions spannerOptions , SessionClient sessionClient ) {
1135
+ static SessionPool createPool (
1136
+ SpannerOptions spannerOptions , SessionClient sessionClient , List <LabelValue > labelValues ) {
1120
1137
return createPool (
1121
1138
spannerOptions .getSessionPoolOptions (),
1122
1139
((GrpcTransportOptions ) spannerOptions .getTransportOptions ()).getExecutorFactory (),
1123
- sessionClient );
1140
+ sessionClient ,
1141
+ new Clock (),
1142
+ Metrics .getMetricRegistry (),
1143
+ labelValues );
1124
1144
}
1125
1145
1126
1146
static SessionPool createPool (
@@ -1135,8 +1155,31 @@ static SessionPool createPool(
1135
1155
ExecutorFactory <ScheduledExecutorService > executorFactory ,
1136
1156
SessionClient sessionClient ,
1137
1157
Clock clock ) {
1158
+ return createPool (
1159
+ poolOptions ,
1160
+ executorFactory ,
1161
+ sessionClient ,
1162
+ clock ,
1163
+ Metrics .getMetricRegistry (),
1164
+ SPANNER_DEFAULT_LABEL_VALUES );
1165
+ }
1166
+
1167
+ static SessionPool createPool (
1168
+ SessionPoolOptions poolOptions ,
1169
+ ExecutorFactory <ScheduledExecutorService > executorFactory ,
1170
+ SessionClient sessionClient ,
1171
+ Clock clock ,
1172
+ MetricRegistry metricRegistry ,
1173
+ List <LabelValue > labelValues ) {
1138
1174
SessionPool pool =
1139
- new SessionPool (poolOptions , executorFactory , executorFactory .get (), sessionClient , clock );
1175
+ new SessionPool (
1176
+ poolOptions ,
1177
+ executorFactory ,
1178
+ executorFactory .get (),
1179
+ sessionClient ,
1180
+ clock ,
1181
+ metricRegistry ,
1182
+ labelValues );
1140
1183
pool .initPool ();
1141
1184
return pool ;
1142
1185
}
@@ -1146,13 +1189,16 @@ private SessionPool(
1146
1189
ExecutorFactory <ScheduledExecutorService > executorFactory ,
1147
1190
ScheduledExecutorService executor ,
1148
1191
SessionClient sessionClient ,
1149
- Clock clock ) {
1192
+ Clock clock ,
1193
+ MetricRegistry metricRegistry ,
1194
+ List <LabelValue > labelValues ) {
1150
1195
this .options = options ;
1151
1196
this .executorFactory = executorFactory ;
1152
1197
this .executor = executor ;
1153
1198
this .sessionClient = sessionClient ;
1154
1199
this .clock = clock ;
1155
1200
this .poolMaintainer = new PoolMaintainer ();
1201
+ this .initMetricsCollection (metricRegistry , labelValues );
1156
1202
}
1157
1203
1158
1204
@ VisibleForTesting
@@ -1766,4 +1812,73 @@ public void onSessionCreateFailure(Throwable t, int createFailureForSessionCount
1766
1812
}
1767
1813
}
1768
1814
}
1815
+
1816
+ /**
1817
+ * Initializes and creates Spanner session relevant metrics. When coupled with an exporter, it
1818
+ * allows users to monitor client behavior.
1819
+ */
1820
+ private void initMetricsCollection (MetricRegistry metricRegistry , List <LabelValue > labelValues ) {
1821
+ DerivedLongGauge maxInUseSessionsMetric =
1822
+ metricRegistry .addDerivedLongGauge (
1823
+ MAX_IN_USE_SESSIONS ,
1824
+ MetricOptions .builder ()
1825
+ .setDescription (MAX_IN_USE_SESSIONS_DESCRIPTION )
1826
+ .setUnit (COUNT )
1827
+ .setLabelKeys (SPANNER_LABEL_KEYS )
1828
+ .build ());
1829
+
1830
+ DerivedLongGauge maxAllowedSessionsMetric =
1831
+ metricRegistry .addDerivedLongGauge (
1832
+ MAX_ALLOWED_SESSIONS ,
1833
+ MetricOptions .builder ()
1834
+ .setDescription (MAX_ALLOWED_SESSIONS_DESCRIPTION )
1835
+ .setUnit (COUNT )
1836
+ .setLabelKeys (SPANNER_LABEL_KEYS )
1837
+ .build ());
1838
+
1839
+ DerivedLongGauge numInUseSessionsMetric =
1840
+ metricRegistry .addDerivedLongGauge (
1841
+ IN_USE_SESSIONS ,
1842
+ MetricOptions .builder ()
1843
+ .setDescription (IN_USE_SESSIONS_DESCRIPTION )
1844
+ .setUnit (COUNT )
1845
+ .setLabelKeys (SPANNER_LABEL_KEYS )
1846
+ .build ());
1847
+
1848
+ // The value of a maxSessionsInUse is observed from a callback function. This function is
1849
+ // invoked whenever metrics are collected.
1850
+ maxInUseSessionsMetric .createTimeSeries (
1851
+ labelValues ,
1852
+ this ,
1853
+ new ToLongFunction <SessionPool >() {
1854
+ @ Override
1855
+ public long applyAsLong (SessionPool sessionPool ) {
1856
+ return sessionPool .maxSessionsInUse ;
1857
+ }
1858
+ });
1859
+
1860
+ // The value of a maxSessions is observed from a callback function. This function is invoked
1861
+ // whenever metrics are collected.
1862
+ maxAllowedSessionsMetric .createTimeSeries (
1863
+ labelValues ,
1864
+ options ,
1865
+ new ToLongFunction <SessionPoolOptions >() {
1866
+ @ Override
1867
+ public long applyAsLong (SessionPoolOptions options ) {
1868
+ return options .getMaxSessions ();
1869
+ }
1870
+ });
1871
+
1872
+ // The value of a numSessionsInUse is observed from a callback function. This function is
1873
+ // invoked whenever metrics are collected.
1874
+ numInUseSessionsMetric .createTimeSeries (
1875
+ labelValues ,
1876
+ this ,
1877
+ new ToLongFunction <SessionPool >() {
1878
+ @ Override
1879
+ public long applyAsLong (SessionPool sessionPool ) {
1880
+ return sessionPool .numSessionsInUse ;
1881
+ }
1882
+ });
1883
+ }
1769
1884
}
0 commit comments