@@ -19,7 +19,6 @@ import 'dart:typed_data';
19
19
20
20
import 'package:collection/collection.dart' ;
21
21
import 'package:grpc/grpc.dart' ;
22
-
23
22
import 'package:interop/src/generated/empty.pb.dart' ;
24
23
import 'package:interop/src/generated/messages.pb.dart' ;
25
24
import 'package:interop/src/generated/test.pbgrpc.dart' ;
@@ -40,6 +39,17 @@ class Tester {
40
39
String defaultServiceAccount;
41
40
String oauthScope;
42
41
String serviceAccountKeyFile;
42
+ String _serviceAccountJson;
43
+
44
+ String get serviceAccountJson =>
45
+ _serviceAccountJson ?? = _readServiceAccountJson ();
46
+
47
+ String _readServiceAccountJson () {
48
+ if (serviceAccountKeyFile? .isEmpty ?? true ) {
49
+ throw 'Service account key file not specified.' ;
50
+ }
51
+ return new File (serviceAccountKeyFile).readAsStringSync ();
52
+ }
43
53
44
54
void set serverPort (String value) {
45
55
if (value == null ) {
@@ -61,6 +71,7 @@ class Tester {
61
71
_useTestCA = value == 'true' ;
62
72
}
63
73
74
+ ClientChannel channel;
64
75
TestServiceClient client;
65
76
UnimplementedServiceClient unimplementedServiceClient;
66
77
@@ -90,7 +101,7 @@ class Tester {
90
101
options = new ChannelOptions .insecure ();
91
102
}
92
103
93
- final channel =
104
+ channel =
94
105
new ClientChannel (serverHost, port: _serverPort, options: options);
95
106
client = new TestServiceClient (channel);
96
107
unimplementedServiceClient = new UnimplementedServiceClient (channel);
@@ -124,6 +135,8 @@ class Tester {
124
135
return emptyStream ();
125
136
case 'compute_engine_creds' :
126
137
return computeEngineCreds ();
138
+ case 'service_account_creds' :
139
+ return serviceAccountCreds ();
127
140
case 'jwt_token_creds' :
128
141
return jwtTokenCreds ();
129
142
case 'oauth2_auth_token' :
@@ -458,7 +471,8 @@ class Tester {
458
471
responses.map ((response) => response.payload.body.length).toList ();
459
472
460
473
if (! new ListEquality ().equals (responseLengths, expectedResponses)) {
461
- throw 'Incorrect response lengths received (${responseLengths .join (', ' )} != ${expectedResponses .join (', ' )})' ;
474
+ throw 'Incorrect response lengths received (${responseLengths .join (
475
+ ', ' )} != ${expectedResponses .join (', ' )})' ;
462
476
}
463
477
}
464
478
@@ -571,7 +585,8 @@ class Tester {
571
585
requests.add (index);
572
586
await for (final response in responses) {
573
587
if (index >= expectedResponses.length) {
574
- throw 'Received too many responses. $index > ${expectedResponses .length }.' ;
588
+ throw 'Received too many responses. $index > ${expectedResponses
589
+ .length }.' ;
575
590
}
576
591
if (response.payload.body.length != expectedResponses[index]) {
577
592
throw 'Response mismatch for response $index : '
@@ -638,6 +653,62 @@ class Tester {
638
653
/// * clients are free to assert that the response payload body contents are
639
654
/// zero and comparing the entire response message against a golden response
640
655
Future <Null > computeEngineCreds () async {
656
+ final credentials = new ComputeEngineAuthenticator ();
657
+ final clientWithCredentials =
658
+ new TestServiceClient (channel, options: credentials.toCallOptions);
659
+
660
+ final response = await _sendSimpleRequestForAuth (clientWithCredentials,
661
+ fillUsername: true , fillOauthScope: true );
662
+
663
+ final user = response.username;
664
+ final oauth = response.oauthScope;
665
+
666
+ if (user? .isEmpty ?? true ) {
667
+ throw 'Username not received.' ;
668
+ }
669
+ if (oauth? .isEmpty ?? true ) {
670
+ throw 'OAuth scope not received.' ;
671
+ }
672
+
673
+ if (! serviceAccountJson.contains (user)) {
674
+ throw 'Got user name $user , which is not a substring of $serviceAccountJson ' ;
675
+ }
676
+ if (! oauthScope.contains (oauth)) {
677
+ throw 'Got OAuth scope $oauth , which is not a substring of $oauthScope ' ;
678
+ }
679
+ }
680
+
681
+ /// This test is only for cloud-to-prod path.
682
+ ///
683
+ /// This test verifies unary calls succeed in sending messages while using
684
+ /// service account credentials.
685
+ ///
686
+ /// Test caller should set flag `--service_account_key_file` with the path to
687
+ /// json key file downloaded from https://console.developers.google.com.
688
+ /// Alternately, if using a usable auth implementation, she may specify the
689
+ /// file location in the environment variable GOOGLE_APPLICATION_CREDENTIALS.
690
+ ///
691
+ /// Procedure:
692
+ /// 1. Client configures the channel to use ServiceAccountCredentials
693
+ /// 2. Client calls UnaryCall with:
694
+ /// {
695
+ /// response_size: 314159
696
+ /// payload: {
697
+ /// body: 271828 bytes of zeros
698
+ /// }
699
+ /// fill_username: true
700
+ /// }
701
+ ///
702
+ /// Client asserts:
703
+ /// * call was successful
704
+ /// * received SimpleResponse.username is not empty and is in the json key
705
+ /// file used by the auth library. The client can optionally check the
706
+ /// username matches the email address in the key file or equals the value
707
+ /// of `--default_service_account` flag.
708
+ /// * response payload body is 314159 bytes in size
709
+ /// * clients are free to assert that the response payload body contents are
710
+ /// zero and comparing the entire response message against a golden response
711
+ Future <Null > serviceAccountCreds () async {
641
712
throw 'Not implemented' ;
642
713
}
643
714
@@ -672,7 +743,19 @@ class Tester {
672
743
/// * clients are free to assert that the response payload body contents are
673
744
/// zero and comparing the entire response message against a golden response
674
745
Future <Null > jwtTokenCreds () async {
675
- throw 'Not implemented' ;
746
+ final credentials = new JwtServiceAccountAuthenticator (serviceAccountJson);
747
+ final clientWithCredentials =
748
+ new TestServiceClient (channel, options: credentials.toCallOptions);
749
+
750
+ final response = await _sendSimpleRequestForAuth (clientWithCredentials,
751
+ fillUsername: true );
752
+ final username = response.username;
753
+ if (username? .isEmpty ?? true ) {
754
+ throw 'Username not received.' ;
755
+ }
756
+ if (! serviceAccountJson.contains (username)) {
757
+ throw 'Got user name $username , which is not a substring of $serviceAccountJson ' ;
758
+ }
676
759
}
677
760
678
761
/// This test is only for cloud-to-prod path and some implementations may run
@@ -715,7 +798,30 @@ class Tester {
715
798
/// check against the json key file or GCE default service account email.
716
799
/// * received SimpleResponse.oauth_scope is in `--oauth_scope`
717
800
Future <Null > oauth2AuthToken () async {
718
- throw 'Not implemented' ;
801
+ final credentials =
802
+ new ServiceAccountAuthenticator (serviceAccountJson, [oauthScope]);
803
+ final clientWithCredentials =
804
+ new TestServiceClient (channel, options: credentials.toCallOptions);
805
+
806
+ final response = await _sendSimpleRequestForAuth (clientWithCredentials,
807
+ fillUsername: true , fillOauthScope: true );
808
+
809
+ final user = response.username;
810
+ final oauth = response.oauthScope;
811
+
812
+ if (user? .isEmpty ?? true ) {
813
+ throw 'Username not received.' ;
814
+ }
815
+ if (oauth? .isEmpty ?? true ) {
816
+ throw 'OAuth scope not received.' ;
817
+ }
818
+
819
+ if (! serviceAccountJson.contains (user)) {
820
+ throw 'Got user name $user , which is not a substring of $serviceAccountJson ' ;
821
+ }
822
+ if (! oauthScope.contains (oauth)) {
823
+ throw 'Got OAuth scope $oauth , which is not a substring of $oauthScope ' ;
824
+ }
719
825
}
720
826
721
827
/// Similar to the other auth tests, this test is only for cloud-to-prod path.
@@ -747,7 +853,49 @@ class Tester {
747
853
/// file used by the auth library. The client can optionally check the
748
854
/// username matches the email address in the key file.
749
855
Future <Null > perRpcCreds () async {
750
- throw 'Not implemented' ;
856
+ final credentials =
857
+ new ServiceAccountAuthenticator (serviceAccountJson, [oauthScope]);
858
+
859
+ final response = await _sendSimpleRequestForAuth (client,
860
+ fillUsername: true ,
861
+ fillOauthScope: true ,
862
+ options: credentials.toCallOptions);
863
+
864
+ final user = response.username;
865
+ final oauth = response.oauthScope;
866
+
867
+ if (user? .isEmpty ?? true ) {
868
+ throw 'Username not received.' ;
869
+ }
870
+ if (oauth? .isEmpty ?? true ) {
871
+ throw 'OAuth scope not received.' ;
872
+ }
873
+
874
+ if (! serviceAccountJson.contains (user)) {
875
+ throw 'Got user name $user , which is not a substring of $serviceAccountJson ' ;
876
+ }
877
+ if (! oauthScope.contains (oauth)) {
878
+ throw 'Got OAuth scope $oauth , which is not a substring of $oauthScope ' ;
879
+ }
880
+ }
881
+
882
+ Future <SimpleResponse > _sendSimpleRequestForAuth (TestServiceClient client,
883
+ {bool fillUsername: false ,
884
+ bool fillOauthScope: false ,
885
+ CallOptions options}) async {
886
+ final payload = new Payload ()..body = new Uint8List (271828 );
887
+ final request = new SimpleRequest ()
888
+ ..responseSize = 314159
889
+ ..payload = payload
890
+ ..fillUsername = fillUsername
891
+ ..fillOauthScope = fillOauthScope;
892
+ final response = await client.unaryCall (request, options: options);
893
+ final receivedBytes = response.payload.body.length;
894
+ if (receivedBytes != 314159 ) {
895
+ throw 'Response payload mismatch. Expected 314159 bytes, '
896
+ 'got ${receivedBytes }.' ;
897
+ }
898
+ return response;
751
899
}
752
900
753
901
/// This test verifies that custom metadata in either binary or ascii format
0 commit comments