@@ -29,7 +29,10 @@ import (
29
29
"time"
30
30
31
31
"cloud.google.com/go/internal/trace"
32
+ "cloud.google.com/go/internal/version"
32
33
vkit "cloud.google.com/go/spanner/apiv1"
34
+ "go.opencensus.io/stats"
35
+ "go.opencensus.io/tag"
33
36
sppb "google.golang.org/genproto/googleapis/spanner/v1"
34
37
"google.golang.org/grpc/codes"
35
38
"google.golang.org/grpc/metadata"
@@ -520,13 +523,23 @@ type sessionPool struct {
520
523
SessionPoolConfig
521
524
// hc is the health checker
522
525
hc * healthChecker
526
+ // rand is a separately sourced random generator.
527
+ rand * rand.Rand
528
+ // numInUse is the number of sessions that are currently in use (checked out
529
+ // from the session pool).
530
+ numInUse uint64
531
+ // maxNumInUse is the maximum number of sessions in use concurrently in the
532
+ // current 10 minute interval.
533
+ maxNumInUse uint64
534
+ // lastResetTime is the start time of the window for recording maxNumInUse.
535
+ lastResetTime time.Time
523
536
524
537
// mw is the maintenance window containing statistics for the max number of
525
538
// sessions checked out of the pool during the last 10 minutes.
526
539
mw * maintenanceWindow
527
540
528
- // rand is a separately sourced random generator .
529
- rand * rand. Rand
541
+ // tagMap is a map of all tags that are associated with the emitted metrics .
542
+ tagMap * tag. Map
530
543
}
531
544
532
545
// newSessionPool creates a new session pool.
@@ -557,6 +570,23 @@ func newSessionPool(sc *sessionClient, config SessionPoolConfig) (*sessionPool,
557
570
if config .healthCheckSampleInterval == 0 {
558
571
config .healthCheckSampleInterval = time .Minute
559
572
}
573
+
574
+ _ , instance , database , err := parseDatabaseName (sc .database )
575
+ if err != nil {
576
+ return nil , err
577
+ }
578
+ // Errors should not prevent initializing the session pool.
579
+ ctx , err := tag .New (context .Background (),
580
+ tag .Upsert (tagClientID , sc .id ),
581
+ tag .Upsert (tagDatabase , database ),
582
+ tag .Upsert (tagInstance , instance ),
583
+ tag .Upsert (tagLibVersion , version .Repo ),
584
+ )
585
+ if err != nil {
586
+ logf (pool .sc .logger , "Failed to create tag map, error: %v" , err )
587
+ }
588
+ pool .tagMap = tag .FromContext (ctx )
589
+
560
590
// On GCE VM, within the same region an healthcheck ping takes on average
561
591
// 10ms to finish, given a 5 minutes interval and 10 healthcheck workers, a
562
592
// healthChecker can effectively mantain
@@ -573,15 +603,21 @@ func newSessionPool(sc *sessionClient, config SessionPoolConfig) (*sessionPool,
573
603
return nil , err
574
604
}
575
605
}
606
+ pool .recordStat (context .Background (), MaxAllowedSessionsCount , int64 (config .MaxOpened ))
576
607
close (pool .hc .ready )
577
608
return pool , nil
578
609
}
579
610
611
+ func (p * sessionPool ) recordStat (ctx context.Context , m * stats.Int64Measure , n int64 ) {
612
+ ctx = tag .NewContext (ctx , p .tagMap )
613
+ recordStat (ctx , m , n )
614
+ }
615
+
580
616
func (p * sessionPool ) initPool (numSessions int32 ) error {
581
617
p .mu .Lock ()
582
618
// Take budget before the actual session creation.
583
619
p .numOpened += uint64 (numSessions )
584
- recordStat (context .Background (), OpenSessionCount , int64 (p .numOpened ))
620
+ p . recordStat (context .Background (), OpenSessionCount , int64 (p .numOpened ))
585
621
p .createReqs += uint64 (numSessions )
586
622
p .mu .Unlock ()
587
623
// Asynchronously create the initial sessions for the pool.
@@ -626,7 +662,7 @@ func (p *sessionPool) sessionCreationFailed(err error, numSessions int32) {
626
662
defer p .mu .Unlock ()
627
663
p .createReqs -= uint64 (numSessions )
628
664
p .numOpened -= uint64 (numSessions )
629
- recordStat (context .Background (), OpenSessionCount , int64 (p .numOpened ))
665
+ p . recordStat (context .Background (), OpenSessionCount , int64 (p .numOpened ))
630
666
// Notify other waiters blocking on session creation.
631
667
close (p .mayGetSession )
632
668
p .mayGetSession = make (chan struct {})
@@ -746,7 +782,7 @@ func (p *sessionPool) createSession(ctx context.Context) (*session, error) {
746
782
if ! done {
747
783
// Session creation failed, give budget back.
748
784
p .numOpened --
749
- recordStat (ctx , OpenSessionCount , int64 (p .numOpened ))
785
+ p . recordStat (ctx , OpenSessionCount , int64 (p .numOpened ))
750
786
}
751
787
p .createReqs --
752
788
// Notify other waiters blocking on session creation.
@@ -823,6 +859,7 @@ func (p *sessionPool) take(ctx context.Context) (*sessionHandle, error) {
823
859
if ! p .isHealthy (s ) {
824
860
continue
825
861
}
862
+ p .incNumInUse (ctx )
826
863
return p .newSessionHandle (s ), nil
827
864
}
828
865
@@ -835,6 +872,7 @@ func (p *sessionPool) take(ctx context.Context) (*sessionHandle, error) {
835
872
select {
836
873
case <- ctx .Done ():
837
874
trace .TracePrintf (ctx , nil , "Context done waiting for session" )
875
+ p .recordStat (ctx , GetSessionTimeoutsCount , 1 )
838
876
return nil , p .errGetSessionTimeout ()
839
877
case <- mayGetSession :
840
878
}
@@ -846,7 +884,7 @@ func (p *sessionPool) take(ctx context.Context) (*sessionHandle, error) {
846
884
// Creating a new session that will be returned directly to the client
847
885
// means that the max number of sessions in use also increases.
848
886
numCheckedOut := p .currSessionsCheckedOutLocked ()
849
- recordStat (ctx , OpenSessionCount , int64 (p .numOpened ))
887
+ p . recordStat (ctx , OpenSessionCount , int64 (p .numOpened ))
850
888
p .createReqs ++
851
889
p .mu .Unlock ()
852
890
p .mw .updateMaxSessionsCheckedOutDuringWindow (numCheckedOut )
@@ -856,6 +894,7 @@ func (p *sessionPool) take(ctx context.Context) (*sessionHandle, error) {
856
894
}
857
895
trace .TracePrintf (ctx , map [string ]interface {}{"sessionID" : s .getID ()},
858
896
"Created session" )
897
+ p .incNumInUse (ctx )
859
898
return p .newSessionHandle (s ), nil
860
899
}
861
900
}
@@ -908,6 +947,7 @@ func (p *sessionPool) takeWriteSession(ctx context.Context) (*sessionHandle, err
908
947
select {
909
948
case <- ctx .Done ():
910
949
trace .TracePrintf (ctx , nil , "Context done waiting for session" )
950
+ p .recordStat (ctx , GetSessionTimeoutsCount , 1 )
911
951
return nil , p .errGetSessionTimeout ()
912
952
case <- mayGetSession :
913
953
}
@@ -919,7 +959,7 @@ func (p *sessionPool) takeWriteSession(ctx context.Context) (*sessionHandle, err
919
959
// Creating a new session that will be returned directly to the client
920
960
// means that the max number of sessions in use also increases.
921
961
numCheckedOut := p .currSessionsCheckedOutLocked ()
922
- recordStat (ctx , OpenSessionCount , int64 (p .numOpened ))
962
+ p . recordStat (ctx , OpenSessionCount , int64 (p .numOpened ))
923
963
p .createReqs ++
924
964
p .mu .Unlock ()
925
965
p .mw .updateMaxSessionsCheckedOutDuringWindow (numCheckedOut )
@@ -945,6 +985,7 @@ func (p *sessionPool) takeWriteSession(ctx context.Context) (*sessionHandle, err
945
985
return nil , toSpannerError (err )
946
986
}
947
987
}
988
+ p .incNumInUse (ctx )
948
989
return p .newSessionHandle (s ), nil
949
990
}
950
991
}
@@ -968,6 +1009,9 @@ func (p *sessionPool) recycle(s *session) bool {
968
1009
// Broadcast that a session has been returned to idle list.
969
1010
close (p .mayGetSession )
970
1011
p .mayGetSession = make (chan struct {})
1012
+ p .numInUse --
1013
+ p .recordStat (context .Background (), InUseSessionsCount , int64 (p .numInUse ))
1014
+ p .recordStat (context .Background (), ReleasedSessionsCount , 1 )
971
1015
return true
972
1016
}
973
1017
@@ -992,7 +1036,7 @@ func (p *sessionPool) remove(s *session, isExpire bool) bool {
992
1036
if s .invalidate () {
993
1037
// Decrease the number of opened sessions.
994
1038
p .numOpened --
995
- recordStat (context .Background (), OpenSessionCount , int64 (p .numOpened ))
1039
+ p . recordStat (context .Background (), OpenSessionCount , int64 (p .numOpened ))
996
1040
// Broadcast that a session has been destroyed.
997
1041
close (p .mayGetSession )
998
1042
p .mayGetSession = make (chan struct {})
@@ -1005,6 +1049,22 @@ func (p *sessionPool) currSessionsCheckedOutLocked() uint64 {
1005
1049
return p .numOpened - uint64 (p .idleList .Len ()) - uint64 (p .idleWriteList .Len ())
1006
1050
}
1007
1051
1052
+ func (p * sessionPool ) incNumInUse (ctx context.Context ) {
1053
+ p .mu .Lock ()
1054
+ p .incNumInUseLocked (ctx )
1055
+ p .mu .Unlock ()
1056
+ }
1057
+
1058
+ func (p * sessionPool ) incNumInUseLocked (ctx context.Context ) {
1059
+ p .numInUse ++
1060
+ p .recordStat (ctx , InUseSessionsCount , int64 (p .numInUse ))
1061
+ p .recordStat (ctx , AcquiredSessionsCount , 1 )
1062
+ if p .numInUse > p .maxNumInUse {
1063
+ p .maxNumInUse = p .numInUse
1064
+ p .recordStat (ctx , MaxInUseSessionsCount , int64 (p .maxNumInUse ))
1065
+ }
1066
+ }
1067
+
1008
1068
// hcHeap implements heap.Interface. It is used to create the priority queue for
1009
1069
// session healthchecks.
1010
1070
type hcHeap struct {
@@ -1305,6 +1365,7 @@ func (hc *healthChecker) worker(i int) {
1305
1365
session := hc .pool .idleList .Remove (hc .pool .idleList .Front ()).(* session )
1306
1366
session .checkingHealth = true
1307
1367
hc .pool .prepareReqs ++
1368
+ hc .pool .incNumInUseLocked (context .Background ())
1308
1369
return session
1309
1370
}
1310
1371
}
@@ -1381,6 +1442,15 @@ func (hc *healthChecker) maintainer() {
1381
1442
currSessionsOpened := hc .pool .numOpened
1382
1443
maxIdle := hc .pool .MaxIdle
1383
1444
minOpened := hc .pool .MinOpened
1445
+
1446
+ // Reset the start time for recording the maximum number of sessions
1447
+ // in the pool.
1448
+ now := time .Now ()
1449
+ if now .After (hc .pool .lastResetTime .Add (10 * time .Minute )) {
1450
+ hc .pool .maxNumInUse = hc .pool .numInUse
1451
+ hc .pool .recordStat (context .Background (), MaxInUseSessionsCount , int64 (hc .pool .maxNumInUse ))
1452
+ hc .pool .lastResetTime = now
1453
+ }
1384
1454
hc .pool .mu .Unlock ()
1385
1455
// Get the maximum number of sessions in use during the current
1386
1456
// maintenance window.
@@ -1438,7 +1508,7 @@ func (hc *healthChecker) growPool(ctx context.Context, growToNumSessions uint64)
1438
1508
break
1439
1509
}
1440
1510
p .numOpened ++
1441
- recordStat (ctx , OpenSessionCount , int64 (p .numOpened ))
1511
+ p . recordStat (ctx , OpenSessionCount , int64 (p .numOpened ))
1442
1512
p .createReqs ++
1443
1513
shouldPrepareWrite := p .shouldPrepareWriteLocked ()
1444
1514
p .mu .Unlock ()
0 commit comments