1 // Copyright 2015 The Weave Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4 
5 #include "src/privet/privet_handler.h"
6 
7 #include <set>
8 #include <string>
9 #include <utility>
10 
11 #include <base/bind.h>
12 #include <base/json/json_reader.h>
13 #include <base/json/json_writer.h>
14 #include <base/strings/string_util.h>
15 #include <base/values.h>
16 #include <gmock/gmock.h>
17 #include <gtest/gtest.h>
18 #include <weave/test/unittest_utils.h>
19 
20 #include "src/privet/constants.h"
21 #include "src/privet/mock_delegates.h"
22 #include "src/test/mock_clock.h"
23 
24 using testing::_;
25 using testing::DoAll;
26 using testing::Invoke;
27 using testing::Return;
28 using testing::SetArgPointee;
29 using testing::SaveArg;
30 using testing::WithArgs;
31 
32 namespace weave {
33 namespace privet {
34 
35 namespace {
36 
LoadTestJson(const std::string & test_json,base::DictionaryValue * dictionary)37 void LoadTestJson(const std::string& test_json,
38                   base::DictionaryValue* dictionary) {
39   std::string json = test_json;
40   base::ReplaceChars(json, "'", "\"", &json);
41   int error = 0;
42   std::string message;
43   std::unique_ptr<base::Value> value(
44       base::JSONReader::ReadAndReturnError(json, base::JSON_PARSE_RFC, &error,
45                                            &message)
46           .release());
47   EXPECT_TRUE(value.get()) << "\nError: " << message << "\n" << json;
48   base::DictionaryValue* dictionary_ptr = nullptr;
49   if (value->GetAsDictionary(&dictionary_ptr))
50     dictionary->MergeDictionary(dictionary_ptr);
51 }
52 
53 struct CodeWithReason {
CodeWithReasonweave::privet::__anondfad59590111::CodeWithReason54   CodeWithReason(int code_in, const std::string& reason_in)
55       : code(code_in), reason(reason_in) {}
56   int code;
57   std::string reason;
58 };
59 
operator <<(std::ostream & stream,const CodeWithReason & error)60 std::ostream& operator<<(std::ostream& stream, const CodeWithReason& error) {
61   return stream << "{" << error.code << ", " << error.reason << "}";
62 }
63 
IsEqualError(const CodeWithReason & expected,const base::DictionaryValue & dictionary)64 bool IsEqualError(const CodeWithReason& expected,
65                   const base::DictionaryValue& dictionary) {
66   std::string reason;
67   int code = 0;
68   return dictionary.GetInteger("error.http_status", &code) &&
69          code == expected.code && dictionary.GetString("error.code", &reason) &&
70          reason == expected.reason;
71 }
72 
73 // Some error sections in response JSON objects contained debugging information
74 // which is of no interest for this test. So, remove the debug info from the
75 // JSON before running validation logic on it.
StripDebugErrorDetails(const std::string & path_to_error_object,const base::DictionaryValue & value)76 std::unique_ptr<base::DictionaryValue> StripDebugErrorDetails(
77     const std::string& path_to_error_object,
78     const base::DictionaryValue& value) {
79   std::unique_ptr<base::DictionaryValue> result{value.DeepCopy()};
80   base::DictionaryValue* error_dict = nullptr;
81   EXPECT_TRUE(result->GetDictionary(path_to_error_object, &error_dict));
82   scoped_ptr<base::Value> dummy;
83   error_dict->RemovePath("error.debugInfo", &dummy);
84   error_dict->RemovePath("error.message", &dummy);
85   return result;
86 }
87 
88 }  // namespace
89 
90 class PrivetHandlerTest : public testing::Test {
91  public:
PrivetHandlerTest()92   PrivetHandlerTest() {}
93 
94  protected:
SetUp()95   void SetUp() override {
96     EXPECT_CALL(clock_, Now())
97         .WillRepeatedly(Return(base::Time::FromTimeT(1410000001)));
98 
99     auth_header_ = "Privet anonymous";
100     handler_.reset(
101         new PrivetHandler(&cloud_, &device_, &security_, &wifi_, &clock_));
102   }
103 
HandleRequest(const std::string & api,const base::DictionaryValue * input)104   const base::DictionaryValue& HandleRequest(
105       const std::string& api,
106       const base::DictionaryValue* input) {
107     output_.Clear();
108     handler_->HandleRequest(api, auth_header_, input,
109                             base::Bind(&PrivetHandlerTest::HandlerCallback,
110                                        base::Unretained(this)));
111     return output_;
112   }
113 
HandleRequest(const std::string & api,const std::string & json_input)114   const base::DictionaryValue& HandleRequest(const std::string& api,
115                                              const std::string& json_input) {
116     base::DictionaryValue dictionary;
117     LoadTestJson(json_input, &dictionary);
118     return HandleRequest(api, &dictionary);
119   }
120 
HandleUnknownRequest(const std::string & api)121   void HandleUnknownRequest(const std::string& api) {
122     output_.Clear();
123     base::DictionaryValue dictionary;
124     handler_->HandleRequest(api, auth_header_, &dictionary,
125                             base::Bind(&PrivetHandlerTest::HandlerNoFound));
126   }
127 
GetResponse() const128   const base::DictionaryValue& GetResponse() const { return output_; }
GetResponseCount() const129   int GetResponseCount() const { return response_count_; }
130 
SetNoWifiAndGcd()131   void SetNoWifiAndGcd() {
132     handler_.reset(
133         new PrivetHandler(&cloud_, &device_, &security_, nullptr, &clock_));
134     EXPECT_CALL(cloud_, GetCloudId()).WillRepeatedly(Return(""));
135     EXPECT_CALL(cloud_, GetConnectionState())
136         .WillRepeatedly(ReturnRef(gcd_disabled_state_));
137     auto set_error = [](const std::string&, const std::string&,
138                         ErrorPtr* error) {
139       Error::AddTo(error, FROM_HERE, "setupUnavailable", "");
140     };
141     EXPECT_CALL(cloud_, Setup(_, _, _))
142         .WillRepeatedly(DoAll(Invoke(set_error), Return(false)));
143   }
144 
145   test::MockClock clock_;
146   testing::StrictMock<MockCloudDelegate> cloud_;
147   testing::StrictMock<MockDeviceDelegate> device_;
148   testing::StrictMock<MockSecurityDelegate> security_;
149   testing::StrictMock<MockWifiDelegate> wifi_;
150   std::string auth_header_;
151 
152  private:
HandlerCallback(int status,const base::DictionaryValue & output)153   void HandlerCallback(int status, const base::DictionaryValue& output) {
154     output_.Clear();
155     ++response_count_;
156     output_.MergeDictionary(&output);
157     if (!output_.HasKey("error")) {
158       EXPECT_EQ(200, status);
159       return;
160     }
161     EXPECT_NE(200, status);
162     output_.SetInteger("error.http_status", status);
163   }
164 
HandlerNoFound(int status,const base::DictionaryValue &)165   static void HandlerNoFound(int status, const base::DictionaryValue&) {
166     EXPECT_EQ(404, status);
167   }
168 
169   std::unique_ptr<PrivetHandler> handler_;
170   base::DictionaryValue output_;
171   int response_count_{0};
172   ConnectionState gcd_disabled_state_{ConnectionState::kDisabled};
173 };
174 
TEST_F(PrivetHandlerTest,UnknownApi)175 TEST_F(PrivetHandlerTest, UnknownApi) {
176   HandleUnknownRequest("/privet/foo");
177 }
178 
TEST_F(PrivetHandlerTest,InvalidFormat)179 TEST_F(PrivetHandlerTest, InvalidFormat) {
180   auth_header_ = "";
181   EXPECT_PRED2(IsEqualError, CodeWithReason(400, "invalidFormat"),
182                HandleRequest("/privet/info", nullptr));
183 }
184 
TEST_F(PrivetHandlerTest,MissingAuth)185 TEST_F(PrivetHandlerTest, MissingAuth) {
186   auth_header_ = "";
187   EXPECT_PRED2(IsEqualError, CodeWithReason(401, "missingAuthorization"),
188                HandleRequest("/privet/info", "{}"));
189 }
190 
TEST_F(PrivetHandlerTest,InvalidAuth)191 TEST_F(PrivetHandlerTest, InvalidAuth) {
192   auth_header_ = "foo";
193   EXPECT_PRED2(IsEqualError, CodeWithReason(401, "invalidAuthorization"),
194                HandleRequest("/privet/info", "{}"));
195 }
196 
TEST_F(PrivetHandlerTest,ExpiredAuth)197 TEST_F(PrivetHandlerTest, ExpiredAuth) {
198   auth_header_ = "Privet 123";
199   EXPECT_CALL(security_, ParseAccessToken(_, _, _))
200       .WillRepeatedly(WithArgs<2>(Invoke([](ErrorPtr* error) {
201         return Error::AddTo(error, FROM_HERE, "authorizationExpired", "");
202       })));
203   EXPECT_PRED2(IsEqualError, CodeWithReason(403, "authorizationExpired"),
204                HandleRequest("/privet/info", "{}"));
205 }
206 
TEST_F(PrivetHandlerTest,InvalidAuthScope)207 TEST_F(PrivetHandlerTest, InvalidAuthScope) {
208   EXPECT_PRED2(IsEqualError, CodeWithReason(403, "invalidAuthorizationScope"),
209                HandleRequest("/privet/v3/setup/start", "{}"));
210 }
211 
TEST_F(PrivetHandlerTest,InfoMinimal)212 TEST_F(PrivetHandlerTest, InfoMinimal) {
213   SetNoWifiAndGcd();
214   EXPECT_CALL(security_, GetPairingTypes())
215       .WillRepeatedly(Return(std::set<PairingType>{}));
216   EXPECT_CALL(security_, GetCryptoTypes())
217       .WillRepeatedly(Return(std::set<CryptoType>{}));
218   EXPECT_CALL(security_, GetAuthTypes())
219       .WillRepeatedly(Return(std::set<AuthType>{}));
220 
221   const char kExpected[] = R"({
222     'version': '3.0',
223     'id': 'TestId',
224     'name': 'TestDevice',
225     'services': [ "developmentBoard" ],
226     'modelManifestId': "ABMID",
227     'basicModelManifest': {
228       'uiDeviceKind': 'developmentBoard',
229       'oemName': 'Chromium',
230       'modelName': 'Brillo'
231     },
232     'endpoints': {
233       'httpPort': 0,
234       'httpUpdatesPort': 0,
235       'httpsPort': 0,
236       'httpsUpdatesPort': 0
237     },
238     'authentication': {
239       'anonymousMaxScope': 'user',
240       'mode': [
241       ],
242       'pairing': [
243       ],
244       'crypto': [
245       ]
246     },
247     'gcd': {
248       'id': '',
249       'status': 'disabled'
250     },
251     'time': 1410000001000.0,
252     'sessionId': 'SessionId'
253   })";
254   EXPECT_JSON_EQ(kExpected, HandleRequest("/privet/info", "{}"));
255 }
256 
TEST_F(PrivetHandlerTest,Info)257 TEST_F(PrivetHandlerTest, Info) {
258   EXPECT_CALL(cloud_, GetDescription())
259       .WillRepeatedly(Return("TestDescription"));
260   EXPECT_CALL(cloud_, GetLocation()).WillRepeatedly(Return("TestLocation"));
261   EXPECT_CALL(device_, GetHttpEnpoint())
262       .WillRepeatedly(Return(std::make_pair(80, 10080)));
263   EXPECT_CALL(device_, GetHttpsEnpoint())
264       .WillRepeatedly(Return(std::make_pair(443, 10443)));
265   EXPECT_CALL(wifi_, GetHostedSsid())
266       .WillRepeatedly(Return("Test_device.BBABCLAprv"));
267 
268   const char kExpected[] = R"({
269     'version': '3.0',
270     'id': 'TestId',
271     'name': 'TestDevice',
272     'description': 'TestDescription',
273     'location': 'TestLocation',
274     'services': [ "developmentBoard" ],
275     'modelManifestId': "ABMID",
276     'basicModelManifest': {
277       'uiDeviceKind': 'developmentBoard',
278       'oemName': 'Chromium',
279       'modelName': 'Brillo'
280     },
281     'endpoints': {
282       'httpPort': 80,
283       'httpUpdatesPort': 10080,
284       'httpsPort': 443,
285       'httpsUpdatesPort': 10443
286     },
287     'authentication': {
288       'anonymousMaxScope': 'none',
289       'mode': [
290         'anonymous',
291         'pairing',
292         'local'
293       ],
294       'pairing': [
295         'pinCode',
296         'embeddedCode'
297       ],
298       'crypto': [
299         'p224_spake2'
300       ]
301     },
302     'wifi': {
303       'capabilities': [
304         '2.4GHz'
305       ],
306       'ssid': 'TestSsid',
307       'hostedSsid': 'Test_device.BBABCLAprv',
308       'status': 'offline'
309     },
310     'gcd': {
311       'id': 'TestCloudId',
312       'status': 'online'
313     },
314     'time': 1410000001000.0,
315     'sessionId': 'SessionId'
316   })";
317   EXPECT_JSON_EQ(kExpected, HandleRequest("/privet/info", "{}"));
318 }
319 
TEST_F(PrivetHandlerTest,PairingStartInvalidParams)320 TEST_F(PrivetHandlerTest, PairingStartInvalidParams) {
321   EXPECT_PRED2(IsEqualError, CodeWithReason(400, "invalidParams"),
322                HandleRequest("/privet/v3/pairing/start",
323                              "{'pairing':'embeddedCode','crypto':'crypto'}"));
324 
325   EXPECT_PRED2(IsEqualError, CodeWithReason(400, "invalidParams"),
326                HandleRequest("/privet/v3/pairing/start",
327                              "{'pairing':'code','crypto':'p224_spake2'}"));
328 }
329 
TEST_F(PrivetHandlerTest,PairingStart)330 TEST_F(PrivetHandlerTest, PairingStart) {
331   EXPECT_JSON_EQ(
332       "{'deviceCommitment': 'testCommitment', 'sessionId': 'testSession'}",
333       HandleRequest("/privet/v3/pairing/start",
334                     "{'pairing': 'embeddedCode', 'crypto': 'p224_spake2'}"));
335 }
336 
TEST_F(PrivetHandlerTest,PairingConfirm)337 TEST_F(PrivetHandlerTest, PairingConfirm) {
338   EXPECT_JSON_EQ(
339       "{'certFingerprint':'testFingerprint','certSignature':'testSignature'}",
340       HandleRequest(
341           "/privet/v3/pairing/confirm",
342           "{'sessionId':'testSession','clientCommitment':'testCommitment'}"));
343 }
344 
TEST_F(PrivetHandlerTest,PairingCancel)345 TEST_F(PrivetHandlerTest, PairingCancel) {
346   EXPECT_JSON_EQ("{}", HandleRequest("/privet/v3/pairing/cancel",
347                                      "{'sessionId': 'testSession'}"));
348 }
349 
TEST_F(PrivetHandlerTest,AuthErrorNoType)350 TEST_F(PrivetHandlerTest, AuthErrorNoType) {
351   EXPECT_PRED2(IsEqualError, CodeWithReason(400, "invalidAuthMode"),
352                HandleRequest("/privet/v3/auth", "{}"));
353 }
354 
TEST_F(PrivetHandlerTest,AuthErrorInvalidType)355 TEST_F(PrivetHandlerTest, AuthErrorInvalidType) {
356   EXPECT_PRED2(IsEqualError, CodeWithReason(400, "invalidAuthMode"),
357                HandleRequest("/privet/v3/auth", "{'mode':'unknown'}"));
358 }
359 
TEST_F(PrivetHandlerTest,AuthErrorNoScope)360 TEST_F(PrivetHandlerTest, AuthErrorNoScope) {
361   EXPECT_PRED2(IsEqualError, CodeWithReason(400, "invalidRequestedScope"),
362                HandleRequest("/privet/v3/auth", "{'mode':'anonymous'}"));
363 }
364 
TEST_F(PrivetHandlerTest,AuthErrorInvalidScope)365 TEST_F(PrivetHandlerTest, AuthErrorInvalidScope) {
366   EXPECT_PRED2(
367       IsEqualError, CodeWithReason(400, "invalidRequestedScope"),
368       HandleRequest("/privet/v3/auth",
369                     "{'mode':'anonymous','requestedScope':'unknown'}"));
370 }
371 
TEST_F(PrivetHandlerTest,AuthErrorAccessDenied)372 TEST_F(PrivetHandlerTest, AuthErrorAccessDenied) {
373   EXPECT_PRED2(IsEqualError, CodeWithReason(403, "accessDenied"),
374                HandleRequest("/privet/v3/auth",
375                              "{'mode':'anonymous','requestedScope':'owner'}"));
376 }
377 
TEST_F(PrivetHandlerTest,AuthErrorInvalidAuthCode)378 TEST_F(PrivetHandlerTest, AuthErrorInvalidAuthCode) {
379   auto set_error = [](ErrorPtr* error) {
380     return Error::AddTo(error, FROM_HERE, "invalidAuthCode", "");
381   };
382   EXPECT_CALL(security_, CreateAccessToken(_, "testToken", _, _, _, _, _))
383       .WillRepeatedly(WithArgs<6>(Invoke(set_error)));
384   const char kInput[] = R"({
385     'mode': 'pairing',
386     'requestedScope': 'user',
387     'authCode': 'testToken'
388   })";
389   EXPECT_PRED2(IsEqualError, CodeWithReason(403, "invalidAuthCode"),
390                HandleRequest("/privet/v3/auth", kInput));
391 }
392 
TEST_F(PrivetHandlerTest,AuthAnonymous)393 TEST_F(PrivetHandlerTest, AuthAnonymous) {
394   const char kExpected[] = R"({
395     'accessToken': 'GuestAccessToken',
396     'expiresIn': 15,
397     'scope': 'viewer',
398     'tokenType': 'Privet'
399   })";
400   EXPECT_JSON_EQ(kExpected,
401                  HandleRequest("/privet/v3/auth",
402                                "{'mode':'anonymous','requestedScope':'auto'}"));
403 }
404 
TEST_F(PrivetHandlerTest,AuthPairing)405 TEST_F(PrivetHandlerTest, AuthPairing) {
406   EXPECT_CALL(security_, CreateAccessToken(_, _, _, _, _, _, _))
407       .WillRepeatedly(DoAll(SetArgPointee<3>("OwnerAccessToken"),
408                             SetArgPointee<4>(AuthScope::kOwner),
409                             SetArgPointee<5>(base::TimeDelta::FromSeconds(15)),
410                             Return(true)));
411   const char kInput[] = R"({
412     'mode': 'pairing',
413     'requestedScope': 'owner',
414     'authCode': 'testToken'
415   })";
416   const char kExpected[] = R"({
417     'accessToken': 'OwnerAccessToken',
418     'expiresIn': 15,
419     'scope': 'owner',
420     'tokenType': 'Privet'
421   })";
422   EXPECT_JSON_EQ(kExpected, HandleRequest("/privet/v3/auth", kInput));
423 }
424 
TEST_F(PrivetHandlerTest,AuthLocalAuto)425 TEST_F(PrivetHandlerTest, AuthLocalAuto) {
426   EXPECT_CALL(security_, CreateAccessToken(_, _, _, _, _, _, _))
427       .WillRepeatedly(DoAll(SetArgPointee<3>("UserAccessToken"),
428                             SetArgPointee<4>(AuthScope::kUser),
429                             SetArgPointee<5>(base::TimeDelta::FromSeconds(15)),
430                             Return(true)));
431   const char kInput[] = R"({
432     'mode': 'local',
433     'requestedScope': 'auto',
434     'authCode': 'localAuthToken'
435   })";
436   const char kExpected[] = R"({
437     'accessToken': 'UserAccessToken',
438     'expiresIn': 15,
439     'scope': 'user',
440     'tokenType': 'Privet'
441   })";
442   EXPECT_JSON_EQ(kExpected, HandleRequest("/privet/v3/auth", kInput));
443 }
444 
TEST_F(PrivetHandlerTest,AuthLocal)445 TEST_F(PrivetHandlerTest, AuthLocal) {
446   EXPECT_CALL(security_, CreateAccessToken(_, _, _, _, _, _, _))
447       .WillRepeatedly(DoAll(SetArgPointee<3>("ManagerAccessToken"),
448                             SetArgPointee<4>(AuthScope::kManager),
449                             SetArgPointee<5>(base::TimeDelta::FromSeconds(15)),
450                             Return(true)));
451   const char kInput[] = R"({
452     'mode': 'local',
453     'requestedScope': 'manager',
454     'authCode': 'localAuthToken'
455   })";
456   const char kExpected[] = R"({
457     'accessToken': 'ManagerAccessToken',
458     'expiresIn': 15,
459     'scope': 'manager',
460     'tokenType': 'Privet'
461   })";
462   EXPECT_JSON_EQ(kExpected, HandleRequest("/privet/v3/auth", kInput));
463 }
464 
TEST_F(PrivetHandlerTest,AuthLocalHighScope)465 TEST_F(PrivetHandlerTest, AuthLocalHighScope) {
466   EXPECT_CALL(security_, CreateAccessToken(_, _, _, _, _, _, _))
467       .WillRepeatedly(DoAll(SetArgPointee<3>("UserAccessToken"),
468                             SetArgPointee<4>(AuthScope::kUser),
469                             SetArgPointee<5>(base::TimeDelta::FromSeconds(1)),
470                             Return(true)));
471   const char kInput[] = R"({
472     'mode': 'local',
473     'requestedScope': 'manager',
474     'authCode': 'localAuthToken'
475   })";
476   EXPECT_PRED2(IsEqualError, CodeWithReason(403, "accessDenied"),
477                HandleRequest("/privet/v3/auth", kInput));
478 }
479 
480 class PrivetHandlerTestWithAuth : public PrivetHandlerTest {
481  public:
SetUp()482   void SetUp() override {
483     PrivetHandlerTest::SetUp();
484     auth_header_ = "Privet 123";
485     EXPECT_CALL(security_, ParseAccessToken(_, _, _))
486         .WillRepeatedly(DoAll(
487             SetArgPointee<1>(UserInfo{AuthScope::kOwner, TestUserId{"1"}}),
488             Return(true)));
489   }
490 };
491 
492 class PrivetHandlerSetupTest : public PrivetHandlerTestWithAuth {};
493 
TEST_F(PrivetHandlerSetupTest,StatusEmpty)494 TEST_F(PrivetHandlerSetupTest, StatusEmpty) {
495   SetNoWifiAndGcd();
496   EXPECT_JSON_EQ("{}", HandleRequest("/privet/v3/setup/status", "{}"));
497 }
498 
TEST_F(PrivetHandlerSetupTest,StatusWifi)499 TEST_F(PrivetHandlerSetupTest, StatusWifi) {
500   wifi_.setup_state_ = SetupState{SetupState::kSuccess};
501 
502   const char kExpected[] = R"({
503     'wifi': {
504         'ssid': 'TestSsid',
505         'status': 'success'
506      }
507   })";
508   EXPECT_JSON_EQ(kExpected, HandleRequest("/privet/v3/setup/status", "{}"));
509 }
510 
TEST_F(PrivetHandlerSetupTest,StatusWifiError)511 TEST_F(PrivetHandlerSetupTest, StatusWifiError) {
512   ErrorPtr error;
513   Error::AddTo(&error, FROM_HERE, "invalidPassphrase", "");
514   wifi_.setup_state_ = SetupState{std::move(error)};
515 
516   const char kExpected[] = R"({
517     'wifi': {
518         'status': 'error',
519         'error': {
520           'code': 'invalidPassphrase'
521         }
522      }
523   })";
524   EXPECT_JSON_EQ(kExpected,
525                  *StripDebugErrorDetails(
526                      "wifi", HandleRequest("/privet/v3/setup/status", "{}")));
527 }
528 
TEST_F(PrivetHandlerSetupTest,StatusGcd)529 TEST_F(PrivetHandlerSetupTest, StatusGcd) {
530   cloud_.setup_state_ = SetupState{SetupState::kSuccess};
531 
532   const char kExpected[] = R"({
533     'gcd': {
534         'id': 'TestCloudId',
535         'status': 'success'
536      }
537   })";
538   EXPECT_JSON_EQ(kExpected, HandleRequest("/privet/v3/setup/status", "{}"));
539 }
540 
TEST_F(PrivetHandlerSetupTest,StatusGcdError)541 TEST_F(PrivetHandlerSetupTest, StatusGcdError) {
542   ErrorPtr error;
543   Error::AddTo(&error, FROM_HERE, "invalidTicket", "");
544   cloud_.setup_state_ = SetupState{std::move(error)};
545 
546   const char kExpected[] = R"({
547     'gcd': {
548         'status': 'error',
549         'error': {
550           'code': 'invalidTicket'
551         }
552      }
553   })";
554   EXPECT_JSON_EQ(kExpected,
555                  *StripDebugErrorDetails(
556                      "gcd", HandleRequest("/privet/v3/setup/status", "{}")));
557 }
558 
TEST_F(PrivetHandlerSetupTest,SetupNameDescriptionLocation)559 TEST_F(PrivetHandlerSetupTest, SetupNameDescriptionLocation) {
560   EXPECT_CALL(cloud_,
561               UpdateDeviceInfo("testName", "testDescription", "testLocation"))
562       .Times(1);
563   const char kInput[] = R"({
564     'name': 'testName',
565     'description': 'testDescription',
566     'location': 'testLocation'
567   })";
568   EXPECT_JSON_EQ("{}", HandleRequest("/privet/v3/setup/start", kInput));
569 }
570 
TEST_F(PrivetHandlerSetupTest,InvalidParams)571 TEST_F(PrivetHandlerSetupTest, InvalidParams) {
572   const char kInputWifi[] = R"({
573     'wifi': {
574       'ssid': ''
575     }
576   })";
577   EXPECT_PRED2(IsEqualError, CodeWithReason(400, "invalidParams"),
578                HandleRequest("/privet/v3/setup/start", kInputWifi));
579 
580   const char kInputRegistration[] = R"({
581     'gcd': {
582       'ticketId': ''
583     }
584   })";
585   EXPECT_PRED2(IsEqualError, CodeWithReason(400, "invalidParams"),
586                HandleRequest("/privet/v3/setup/start", kInputRegistration));
587 }
588 
TEST_F(PrivetHandlerSetupTest,WifiSetupUnavailable)589 TEST_F(PrivetHandlerSetupTest, WifiSetupUnavailable) {
590   SetNoWifiAndGcd();
591   EXPECT_PRED2(IsEqualError, CodeWithReason(400, "setupUnavailable"),
592                HandleRequest("/privet/v3/setup/start", "{'wifi': {}}"));
593 }
594 
TEST_F(PrivetHandlerSetupTest,WifiSetup)595 TEST_F(PrivetHandlerSetupTest, WifiSetup) {
596   const char kInput[] = R"({
597     'wifi': {
598       'ssid': 'testSsid',
599       'passphrase': 'testPass'
600     }
601   })";
602   auto set_error = [](const std::string&, const std::string&, ErrorPtr* error) {
603     return Error::AddTo(error, FROM_HERE, "deviceBusy", "");
604   };
605   EXPECT_CALL(wifi_, ConfigureCredentials(_, _, _)).WillOnce(Invoke(set_error));
606   EXPECT_PRED2(IsEqualError, CodeWithReason(503, "deviceBusy"),
607                HandleRequest("/privet/v3/setup/start", kInput));
608 
609   const char kExpected[] = R"({
610     'wifi': {
611       'status': 'inProgress'
612     }
613   })";
614   wifi_.setup_state_ = SetupState{SetupState::kInProgress};
615   EXPECT_CALL(wifi_, ConfigureCredentials("testSsid", "testPass", _))
616       .WillOnce(Return(true));
617   EXPECT_JSON_EQ(kExpected, HandleRequest("/privet/v3/setup/start", kInput));
618 }
619 
TEST_F(PrivetHandlerSetupTest,GcdSetupUnavailable)620 TEST_F(PrivetHandlerSetupTest, GcdSetupUnavailable) {
621   SetNoWifiAndGcd();
622   const char kInput[] = R"({
623     'gcd': {
624       'ticketId': 'testTicket',
625       'user': 'testUser'
626     }
627   })";
628 
629   EXPECT_PRED2(IsEqualError, CodeWithReason(400, "setupUnavailable"),
630                HandleRequest("/privet/v3/setup/start", kInput));
631 }
632 
TEST_F(PrivetHandlerSetupTest,GcdSetup)633 TEST_F(PrivetHandlerSetupTest, GcdSetup) {
634   const char kInput[] = R"({
635     'gcd': {
636       'ticketId': 'testTicket',
637       'user': 'testUser'
638     }
639   })";
640 
641   auto set_error = [](const std::string&, const std::string&, ErrorPtr* error) {
642     return Error::AddTo(error, FROM_HERE, "deviceBusy", "");
643   };
644   EXPECT_CALL(cloud_, Setup(_, _, _)).WillOnce(Invoke(set_error));
645   EXPECT_PRED2(IsEqualError, CodeWithReason(503, "deviceBusy"),
646                HandleRequest("/privet/v3/setup/start", kInput));
647 
648   const char kExpected[] = R"({
649     'gcd': {
650       'status': 'inProgress'
651     }
652   })";
653   cloud_.setup_state_ = SetupState{SetupState::kInProgress};
654   EXPECT_CALL(cloud_, Setup("testTicket", "testUser", _))
655       .WillOnce(Return(true));
656   EXPECT_JSON_EQ(kExpected, HandleRequest("/privet/v3/setup/start", kInput));
657 }
658 
TEST_F(PrivetHandlerSetupTest,GcdSetupAsMaster)659 TEST_F(PrivetHandlerSetupTest, GcdSetupAsMaster) {
660   EXPECT_CALL(security_, ParseAccessToken(_, _, _))
661       .WillRepeatedly(DoAll(
662           SetArgPointee<1>(UserInfo{AuthScope::kManager, TestUserId{"1"}}),
663           Return(true)));
664   const char kInput[] = R"({
665     'gcd': {
666       'ticketId': 'testTicket',
667       'user': 'testUser'
668     }
669   })";
670 
671   EXPECT_PRED2(IsEqualError, CodeWithReason(403, "invalidAuthorizationScope"),
672                HandleRequest("/privet/v3/setup/start", kInput));
673 }
674 
TEST_F(PrivetHandlerTestWithAuth,ClaimAccessControl)675 TEST_F(PrivetHandlerTestWithAuth, ClaimAccessControl) {
676   EXPECT_JSON_EQ("{'clientToken': 'RootClientAuthToken'}",
677                  HandleRequest("/privet/v3/accessControl/claim", "{}"));
678 }
679 
TEST_F(PrivetHandlerTestWithAuth,ConfirmAccessControl)680 TEST_F(PrivetHandlerTestWithAuth, ConfirmAccessControl) {
681   EXPECT_JSON_EQ("{}",
682                  HandleRequest("/privet/v3/accessControl/confirm",
683                                "{'clientToken': 'DerivedClientAuthToken'}"));
684 }
685 
TEST_F(PrivetHandlerTestWithAuth,State)686 TEST_F(PrivetHandlerTestWithAuth, State) {
687   EXPECT_JSON_EQ("{'state': {'test': {}}, 'fingerprint': '1'}",
688                  HandleRequest("/privet/v3/state", "{}"));
689 
690   cloud_.NotifyOnStateChanged();
691 
692   EXPECT_JSON_EQ("{'state': {'test': {}}, 'fingerprint': '2'}",
693                  HandleRequest("/privet/v3/state", "{}"));
694 }
695 
TEST_F(PrivetHandlerTestWithAuth,CommandsDefs)696 TEST_F(PrivetHandlerTestWithAuth, CommandsDefs) {
697   EXPECT_JSON_EQ("{'commands': {'test':{}}, 'fingerprint': '1'}",
698                  HandleRequest("/privet/v3/commandDefs", "{}"));
699 
700   cloud_.NotifyOnTraitDefsChanged();
701 
702   EXPECT_JSON_EQ("{'commands': {'test':{}}, 'fingerprint': '2'}",
703                  HandleRequest("/privet/v3/commandDefs", "{}"));
704 }
705 
TEST_F(PrivetHandlerTestWithAuth,Traits)706 TEST_F(PrivetHandlerTestWithAuth, Traits) {
707   EXPECT_JSON_EQ("{'traits': {'test': {}}, 'fingerprint': '1'}",
708                  HandleRequest("/privet/v3/traits", "{}"));
709 
710   cloud_.NotifyOnTraitDefsChanged();
711 
712   EXPECT_JSON_EQ("{'traits': {'test': {}}, 'fingerprint': '2'}",
713                  HandleRequest("/privet/v3/traits", "{}"));
714 }
715 
TEST_F(PrivetHandlerTestWithAuth,Components)716 TEST_F(PrivetHandlerTestWithAuth, Components) {
717   EXPECT_JSON_EQ("{'components': {'test': {}}, 'fingerprint': '1'}",
718                  HandleRequest("/privet/v3/components", "{}"));
719 
720   cloud_.NotifyOnComponentTreeChanged();
721 
722   EXPECT_JSON_EQ("{'components': {'test': {}}, 'fingerprint': '2'}",
723                  HandleRequest("/privet/v3/components", "{}"));
724 
725   // State change will also change the components fingerprint.
726   cloud_.NotifyOnStateChanged();
727 
728   EXPECT_JSON_EQ("{'components': {'test': {}}, 'fingerprint': '3'}",
729                  HandleRequest("/privet/v3/components", "{}"));
730 }
731 
TEST_F(PrivetHandlerTestWithAuth,ComponentsWithFiltersAndPaths)732 TEST_F(PrivetHandlerTestWithAuth, ComponentsWithFiltersAndPaths) {
733   const char kComponents[] = R"({
734     "comp1": {
735       "traits": ["a", "b"],
736       "state": {
737         "a" : {
738           "prop": 1
739         }
740       },
741       "components": {
742         "comp2": {
743           "traits": ["c"],
744           "components": {
745             "comp4": {
746               "traits": ["d"]
747             }
748           }
749         },
750         "comp3": {
751           "traits": ["e"]
752         }
753       }
754     }
755   })";
756   base::DictionaryValue components;
757   LoadTestJson(kComponents, &components);
758   EXPECT_CALL(cloud_, FindComponent(_, _)).WillRepeatedly(Return(nullptr));
759   EXPECT_CALL(cloud_, GetComponents()).WillRepeatedly(ReturnRef(components));
760   const char kExpected1[] = R"({
761     "components": {
762       "comp1": {
763         "state": {
764           "a" : {
765             "prop": 1
766           }
767         }
768       }
769     },
770     "fingerprint": "1"
771   })";
772   EXPECT_JSON_EQ(kExpected1, HandleRequest("/privet/v3/components",
773                                            "{'filter':['state']}"));
774 
775   const char kExpected2[] = R"({
776     "components": {
777       "comp1": {
778         "traits": ["a", "b"]
779       }
780     },
781     "fingerprint": "1"
782   })";
783   EXPECT_JSON_EQ(kExpected2, HandleRequest("/privet/v3/components",
784                                            "{'filter':['traits']}"));
785 
786   const char kExpected3[] = R"({
787     "components": {
788       "comp1": {
789         "components": {
790           "comp2": {
791             "components": {
792               "comp4": {}
793             }
794           },
795           "comp3": {}
796         }
797       }
798     },
799     "fingerprint": "1"
800   })";
801   EXPECT_JSON_EQ(kExpected3, HandleRequest("/privet/v3/components",
802                                            "{'filter':['components']}"));
803 
804   const char kExpected4[] = R"({
805     "components": {
806       "comp1": {
807         "traits": ["a", "b"],
808         "state": {
809           "a" : {
810             "prop": 1
811           }
812         },
813         "components": {
814           "comp2": {
815             "traits": ["c"],
816             "components": {
817               "comp4": {
818                 "traits": ["d"]
819               }
820             }
821           },
822           "comp3": {
823             "traits": ["e"]
824           }
825         }
826       }
827     },
828     "fingerprint": "1"
829   })";
830   EXPECT_JSON_EQ(kExpected4,
831                  HandleRequest("/privet/v3/components",
832                                "{'filter':['traits', 'components', 'state']}"));
833 
834   const base::DictionaryValue* comp2 = nullptr;
835   ASSERT_TRUE(components.GetDictionary("comp1.components.comp2", &comp2));
836   EXPECT_CALL(cloud_, FindComponent("comp1.comp2", _)).WillOnce(Return(comp2));
837 
838   const char kExpected5[] = R"({
839     "components": {
840       "comp2": {
841         "traits": ["c"],
842         "components": {
843           "comp4": {
844             "traits": ["d"]
845           }
846         }
847       }
848     },
849     "fingerprint": "1"
850   })";
851   EXPECT_JSON_EQ(
852       kExpected5,
853       HandleRequest(
854           "/privet/v3/components",
855           "{'path':'comp1.comp2', 'filter':['traits', 'components']}"));
856 
857   auto error_handler = [](ErrorPtr* error) -> const base::DictionaryValue* {
858     return Error::AddTo(error, FROM_HERE, "componentNotFound", "");
859   };
860   EXPECT_CALL(cloud_, FindComponent("comp7", _))
861       .WillOnce(WithArgs<1>(Invoke(error_handler)));
862 
863   EXPECT_PRED2(
864       IsEqualError, CodeWithReason(500, "componentNotFound"),
865       HandleRequest("/privet/v3/components",
866                     "{'path':'comp7', 'filter':['traits', 'components']}"));
867 }
868 
TEST_F(PrivetHandlerTestWithAuth,CommandsExecute)869 TEST_F(PrivetHandlerTestWithAuth, CommandsExecute) {
870   const char kInput[] = "{'name': 'test'}";
871   base::DictionaryValue command;
872   LoadTestJson(kInput, &command);
873   LoadTestJson("{'id':'5'}", &command);
874   EXPECT_CALL(cloud_, AddCommand(_, _, _))
875       .WillOnce(WithArgs<2>(Invoke(
876           [&command](const CloudDelegate::CommandDoneCallback& callback) {
877             callback.Run(command, nullptr);
878           })));
879 
880   EXPECT_JSON_EQ("{'name':'test', 'id':'5'}",
881                  HandleRequest("/privet/v3/commands/execute", kInput));
882 }
883 
TEST_F(PrivetHandlerTestWithAuth,CommandsStatus)884 TEST_F(PrivetHandlerTestWithAuth, CommandsStatus) {
885   const char kInput[] = "{'id': '5'}";
886   base::DictionaryValue command;
887   LoadTestJson(kInput, &command);
888   LoadTestJson("{'name':'test'}", &command);
889   EXPECT_CALL(cloud_, GetCommand(_, _, _))
890       .WillOnce(WithArgs<2>(Invoke(
891           [&command](const CloudDelegate::CommandDoneCallback& callback) {
892             callback.Run(command, nullptr);
893           })));
894 
895   EXPECT_JSON_EQ("{'name':'test', 'id':'5'}",
896                  HandleRequest("/privet/v3/commands/status", kInput));
897 
898   ErrorPtr error;
899   Error::AddTo(&error, FROM_HERE, "notFound", "");
900   EXPECT_CALL(cloud_, GetCommand(_, _, _))
901       .WillOnce(WithArgs<2>(
902           Invoke([&error](const CloudDelegate::CommandDoneCallback& callback) {
903             callback.Run({}, std::move(error));
904           })));
905 
906   EXPECT_PRED2(IsEqualError, CodeWithReason(404, "notFound"),
907                HandleRequest("/privet/v3/commands/status", "{'id': '15'}"));
908 }
909 
TEST_F(PrivetHandlerTestWithAuth,CommandsCancel)910 TEST_F(PrivetHandlerTestWithAuth, CommandsCancel) {
911   const char kExpected[] = "{'id': '5', 'name':'test', 'state':'cancelled'}";
912   base::DictionaryValue command;
913   LoadTestJson(kExpected, &command);
914   EXPECT_CALL(cloud_, CancelCommand(_, _, _))
915       .WillOnce(WithArgs<2>(Invoke(
916           [&command](const CloudDelegate::CommandDoneCallback& callback) {
917             callback.Run(command, nullptr);
918           })));
919 
920   EXPECT_JSON_EQ(kExpected,
921                  HandleRequest("/privet/v3/commands/cancel", "{'id': '8'}"));
922 
923   ErrorPtr error;
924   Error::AddTo(&error, FROM_HERE, "notFound", "");
925   EXPECT_CALL(cloud_, CancelCommand(_, _, _))
926       .WillOnce(WithArgs<2>(
927           Invoke([&error](const CloudDelegate::CommandDoneCallback& callback) {
928             callback.Run({}, std::move(error));
929           })));
930 
931   EXPECT_PRED2(IsEqualError, CodeWithReason(404, "notFound"),
932                HandleRequest("/privet/v3/commands/cancel", "{'id': '11'}"));
933 }
934 
TEST_F(PrivetHandlerTestWithAuth,CommandsList)935 TEST_F(PrivetHandlerTestWithAuth, CommandsList) {
936   const char kExpected[] = R"({
937     'commands' : [
938         {'id':'5', 'state':'cancelled'},
939         {'id':'15', 'state':'inProgress'}
940      ]})";
941 
942   base::DictionaryValue commands;
943   LoadTestJson(kExpected, &commands);
944 
945   EXPECT_CALL(cloud_, ListCommands(_, _))
946       .WillOnce(WithArgs<1>(Invoke(
947           [&commands](const CloudDelegate::CommandDoneCallback& callback) {
948             callback.Run(commands, nullptr);
949           })));
950 
951   EXPECT_JSON_EQ(kExpected, HandleRequest("/privet/v3/commands/list", "{}"));
952 }
953 
954 class PrivetHandlerCheckForUpdatesTest : public PrivetHandlerTestWithAuth {};
955 
TEST_F(PrivetHandlerCheckForUpdatesTest,NoInput)956 TEST_F(PrivetHandlerCheckForUpdatesTest, NoInput) {
957   EXPECT_CALL(device_, GetHttpRequestTimeout())
958       .WillOnce(Return(base::TimeDelta::Max()));
959   cloud_.NotifyOnTraitDefsChanged();
960   cloud_.NotifyOnComponentTreeChanged();
961   cloud_.NotifyOnStateChanged();
962   const char kInput[] = "{}";
963   const char kExpected[] = R"({
964    'commandsFingerprint': '2',
965    'stateFingerprint': '2',
966    'traitsFingerprint': '2',
967    'componentsFingerprint': '3'
968   })";
969   EXPECT_JSON_EQ(kExpected,
970                  HandleRequest("/privet/v3/checkForUpdates", kInput));
971   EXPECT_EQ(1, GetResponseCount());
972 }
973 
TEST_F(PrivetHandlerCheckForUpdatesTest,AlreadyChanged)974 TEST_F(PrivetHandlerCheckForUpdatesTest, AlreadyChanged) {
975   EXPECT_CALL(device_, GetHttpRequestTimeout())
976       .WillOnce(Return(base::TimeDelta::Max()));
977   cloud_.NotifyOnTraitDefsChanged();
978   cloud_.NotifyOnComponentTreeChanged();
979   cloud_.NotifyOnStateChanged();
980   const char kInput[] = R"({
981    'commandsFingerprint': '1',
982    'stateFingerprint': '1',
983    'traitsFingerprint': '1',
984    'componentsFingerprint': '1'
985   })";
986   const char kExpected[] = R"({
987    'commandsFingerprint': '2',
988    'stateFingerprint': '2',
989    'traitsFingerprint': '2',
990    'componentsFingerprint': '3'
991   })";
992   EXPECT_JSON_EQ(kExpected,
993                  HandleRequest("/privet/v3/checkForUpdates", kInput));
994   EXPECT_EQ(1, GetResponseCount());
995 }
996 
TEST_F(PrivetHandlerCheckForUpdatesTest,LongPollCommands)997 TEST_F(PrivetHandlerCheckForUpdatesTest, LongPollCommands) {
998   EXPECT_CALL(device_, GetHttpRequestTimeout())
999       .WillOnce(Return(base::TimeDelta::Max()));
1000   const char kInput[] = R"({
1001    'commandsFingerprint': '1',
1002    'stateFingerprint': '1',
1003    'traitsFingerprint': '1',
1004    'componentsFingerprint': '1'
1005   })";
1006   EXPECT_JSON_EQ("{}", HandleRequest("/privet/v3/checkForUpdates", kInput));
1007   EXPECT_EQ(0, GetResponseCount());
1008   cloud_.NotifyOnTraitDefsChanged();
1009   EXPECT_EQ(1, GetResponseCount());
1010   const char kExpected[] = R"({
1011    'commandsFingerprint': '2',
1012    'stateFingerprint': '1',
1013    'traitsFingerprint': '2',
1014    'componentsFingerprint': '1'
1015   })";
1016   EXPECT_JSON_EQ(kExpected, GetResponse());
1017 }
1018 
TEST_F(PrivetHandlerCheckForUpdatesTest,LongPollTraits)1019 TEST_F(PrivetHandlerCheckForUpdatesTest, LongPollTraits) {
1020   EXPECT_CALL(device_, GetHttpRequestTimeout())
1021       .WillOnce(Return(base::TimeDelta::Max()));
1022   const char kInput[] = R"({
1023    'commandsFingerprint': '1',
1024    'stateFingerprint': '1',
1025    'traitsFingerprint': '1',
1026    'componentsFingerprint': '1'
1027   })";
1028   EXPECT_JSON_EQ("{}", HandleRequest("/privet/v3/checkForUpdates", kInput));
1029   EXPECT_EQ(0, GetResponseCount());
1030   cloud_.NotifyOnTraitDefsChanged();
1031   EXPECT_EQ(1, GetResponseCount());
1032   const char kExpected[] = R"({
1033    'commandsFingerprint': '2',
1034    'stateFingerprint': '1',
1035    'traitsFingerprint': '2',
1036    'componentsFingerprint': '1'
1037   })";
1038   EXPECT_JSON_EQ(kExpected, GetResponse());
1039 }
1040 
TEST_F(PrivetHandlerCheckForUpdatesTest,LongPollState)1041 TEST_F(PrivetHandlerCheckForUpdatesTest, LongPollState) {
1042   EXPECT_CALL(device_, GetHttpRequestTimeout())
1043       .WillOnce(Return(base::TimeDelta::Max()));
1044   const char kInput[] = R"({
1045    'commandsFingerprint': '1',
1046    'stateFingerprint': '1',
1047    'traitsFingerprint': '1',
1048    'componentsFingerprint': '1'
1049   })";
1050   EXPECT_JSON_EQ("{}", HandleRequest("/privet/v3/checkForUpdates", kInput));
1051   EXPECT_EQ(0, GetResponseCount());
1052   cloud_.NotifyOnStateChanged();
1053   EXPECT_EQ(1, GetResponseCount());
1054   const char kExpected[] = R"({
1055    'commandsFingerprint': '1',
1056    'stateFingerprint': '2',
1057    'traitsFingerprint': '1',
1058    'componentsFingerprint': '2'
1059   })";
1060   EXPECT_JSON_EQ(kExpected, GetResponse());
1061 }
1062 
TEST_F(PrivetHandlerCheckForUpdatesTest,LongPollComponents)1063 TEST_F(PrivetHandlerCheckForUpdatesTest, LongPollComponents) {
1064   EXPECT_CALL(device_, GetHttpRequestTimeout())
1065       .WillOnce(Return(base::TimeDelta::Max()));
1066   const char kInput[] = R"({
1067    'commandsFingerprint': '1',
1068    'stateFingerprint': '1',
1069    'traitsFingerprint': '1',
1070    'componentsFingerprint': '1'
1071   })";
1072   EXPECT_JSON_EQ("{}", HandleRequest("/privet/v3/checkForUpdates", kInput));
1073   EXPECT_EQ(0, GetResponseCount());
1074   cloud_.NotifyOnComponentTreeChanged();
1075   EXPECT_EQ(1, GetResponseCount());
1076   const char kExpected[] = R"({
1077    'commandsFingerprint': '1',
1078    'stateFingerprint': '1',
1079    'traitsFingerprint': '1',
1080    'componentsFingerprint': '2'
1081   })";
1082   EXPECT_JSON_EQ(kExpected, GetResponse());
1083 }
1084 
TEST_F(PrivetHandlerCheckForUpdatesTest,LongPollIgnoreTraits)1085 TEST_F(PrivetHandlerCheckForUpdatesTest, LongPollIgnoreTraits) {
1086   EXPECT_CALL(device_, GetHttpRequestTimeout())
1087       .WillOnce(Return(base::TimeDelta::Max()));
1088   const char kInput[] = R"({
1089    'stateFingerprint': '1',
1090    'componentsFingerprint': '1'
1091   })";
1092   EXPECT_JSON_EQ("{}", HandleRequest("/privet/v3/checkForUpdates", kInput));
1093   EXPECT_EQ(0, GetResponseCount());
1094   cloud_.NotifyOnTraitDefsChanged();
1095   EXPECT_EQ(0, GetResponseCount());
1096   cloud_.NotifyOnComponentTreeChanged();
1097   EXPECT_EQ(1, GetResponseCount());
1098   const char kExpected[] = R"({
1099    'commandsFingerprint': '2',
1100    'stateFingerprint': '1',
1101    'traitsFingerprint': '2',
1102    'componentsFingerprint': '2'
1103   })";
1104   EXPECT_JSON_EQ(kExpected, GetResponse());
1105 }
1106 
TEST_F(PrivetHandlerCheckForUpdatesTest,LongPollIgnoreState)1107 TEST_F(PrivetHandlerCheckForUpdatesTest, LongPollIgnoreState) {
1108   EXPECT_CALL(device_, GetHttpRequestTimeout())
1109       .WillOnce(Return(base::TimeDelta::Max()));
1110   const char kInput[] = R"({
1111    'commandsFingerprint': '1',
1112    'traitsFingerprint': '1'
1113   })";
1114   EXPECT_JSON_EQ("{}", HandleRequest("/privet/v3/checkForUpdates", kInput));
1115   EXPECT_EQ(0, GetResponseCount());
1116   cloud_.NotifyOnStateChanged();
1117   EXPECT_EQ(0, GetResponseCount());
1118   cloud_.NotifyOnComponentTreeChanged();
1119   EXPECT_EQ(0, GetResponseCount());
1120   cloud_.NotifyOnTraitDefsChanged();
1121   EXPECT_EQ(1, GetResponseCount());
1122   const char kExpected[] = R"({
1123    'commandsFingerprint': '2',
1124    'stateFingerprint': '2',
1125    'traitsFingerprint': '2',
1126    'componentsFingerprint': '3'
1127   })";
1128   EXPECT_JSON_EQ(kExpected, GetResponse());
1129 }
1130 
TEST_F(PrivetHandlerCheckForUpdatesTest,InstantTimeout)1131 TEST_F(PrivetHandlerCheckForUpdatesTest, InstantTimeout) {
1132   EXPECT_CALL(device_, GetHttpRequestTimeout())
1133       .WillOnce(Return(base::TimeDelta::Max()));
1134   const char kInput[] = R"({
1135    'commandsFingerprint': '1',
1136    'stateFingerprint': '1',
1137    'traitsFingerprint': '1',
1138    'componentsFingerprint': '1',
1139    'waitTimeout': 0
1140   })";
1141   const char kExpected[] = R"({
1142    'commandsFingerprint': '1',
1143    'stateFingerprint': '1',
1144    'traitsFingerprint': '1',
1145    'componentsFingerprint': '1'
1146   })";
1147   EXPECT_JSON_EQ(kExpected,
1148                  HandleRequest("/privet/v3/checkForUpdates", kInput));
1149 }
1150 
TEST_F(PrivetHandlerCheckForUpdatesTest,UserTimeout)1151 TEST_F(PrivetHandlerCheckForUpdatesTest, UserTimeout) {
1152   EXPECT_CALL(device_, GetHttpRequestTimeout())
1153       .WillOnce(Return(base::TimeDelta::Max()));
1154   const char kInput[] = R"({
1155    'commandsFingerprint': '1',
1156    'stateFingerprint': '1',
1157    'traitsFingerprint': '1',
1158    'componentsFingerprint': '1',
1159    'waitTimeout': 3
1160   })";
1161   base::Closure callback;
1162   EXPECT_CALL(device_, PostDelayedTask(_, _, base::TimeDelta::FromSeconds(3)))
1163       .WillOnce(SaveArg<1>(&callback));
1164   EXPECT_JSON_EQ("{}", HandleRequest("/privet/v3/checkForUpdates", kInput));
1165   EXPECT_EQ(0, GetResponseCount());
1166   callback.Run();
1167   EXPECT_EQ(1, GetResponseCount());
1168   const char kExpected[] = R"({
1169    'commandsFingerprint': '1',
1170    'stateFingerprint': '1',
1171    'traitsFingerprint': '1',
1172    'componentsFingerprint': '1'
1173   })";
1174   EXPECT_JSON_EQ(kExpected, GetResponse());
1175 }
1176 
TEST_F(PrivetHandlerCheckForUpdatesTest,ServerTimeout)1177 TEST_F(PrivetHandlerCheckForUpdatesTest, ServerTimeout) {
1178   EXPECT_CALL(device_, GetHttpRequestTimeout())
1179       .WillOnce(Return(base::TimeDelta::FromMinutes(1)));
1180   const char kInput[] = R"({
1181    'commandsFingerprint': '1',
1182    'stateFingerprint': '1',
1183    'traitsFingerprint': '1',
1184    'componentsFingerprint': '1'
1185   })";
1186   base::Closure callback;
1187   EXPECT_CALL(device_, PostDelayedTask(_, _, base::TimeDelta::FromSeconds(50)))
1188       .WillOnce(SaveArg<1>(&callback));
1189   EXPECT_JSON_EQ("{}", HandleRequest("/privet/v3/checkForUpdates", kInput));
1190   EXPECT_EQ(0, GetResponseCount());
1191   callback.Run();
1192   EXPECT_EQ(1, GetResponseCount());
1193   const char kExpected[] = R"({
1194    'commandsFingerprint': '1',
1195    'stateFingerprint': '1',
1196    'traitsFingerprint': '1',
1197    'componentsFingerprint': '1'
1198   })";
1199   EXPECT_JSON_EQ(kExpected, GetResponse());
1200 }
1201 
TEST_F(PrivetHandlerCheckForUpdatesTest,VeryShortServerTimeout)1202 TEST_F(PrivetHandlerCheckForUpdatesTest, VeryShortServerTimeout) {
1203   EXPECT_CALL(device_, GetHttpRequestTimeout())
1204       .WillOnce(Return(base::TimeDelta::FromSeconds(5)));
1205   const char kInput[] = R"({
1206    'commandsFingerprint': '1',
1207    'stateFingerprint': '1',
1208    'traitsFingerprint': '1',
1209    'componentsFingerprint': '1'
1210   })";
1211   EXPECT_JSON_EQ(kInput, HandleRequest("/privet/v3/checkForUpdates", kInput));
1212   EXPECT_EQ(1, GetResponseCount());
1213 }
1214 
TEST_F(PrivetHandlerCheckForUpdatesTest,ServerAndUserTimeout)1215 TEST_F(PrivetHandlerCheckForUpdatesTest, ServerAndUserTimeout) {
1216   EXPECT_CALL(device_, GetHttpRequestTimeout())
1217       .WillOnce(Return(base::TimeDelta::FromMinutes(1)));
1218   const char kInput[] = R"({
1219    'commandsFingerprint': '1',
1220    'stateFingerprint': '1',
1221    'traitsFingerprint': '1',
1222    'componentsFingerprint': '1',
1223    'waitTimeout': 10
1224   })";
1225   base::Closure callback;
1226   EXPECT_CALL(device_, PostDelayedTask(_, _, base::TimeDelta::FromSeconds(10)))
1227       .WillOnce(SaveArg<1>(&callback));
1228   EXPECT_JSON_EQ("{}", HandleRequest("/privet/v3/checkForUpdates", kInput));
1229   EXPECT_EQ(0, GetResponseCount());
1230   callback.Run();
1231   EXPECT_EQ(1, GetResponseCount());
1232   const char kExpected[] = R"({
1233    'commandsFingerprint': '1',
1234    'stateFingerprint': '1',
1235    'traitsFingerprint': '1',
1236    'componentsFingerprint': '1'
1237   })";
1238   EXPECT_JSON_EQ(kExpected, GetResponse());
1239 }
1240 
TEST_F(PrivetHandlerCheckForUpdatesTest,ChangeBeforeTimeout)1241 TEST_F(PrivetHandlerCheckForUpdatesTest, ChangeBeforeTimeout) {
1242   EXPECT_CALL(device_, GetHttpRequestTimeout())
1243       .WillOnce(Return(base::TimeDelta::Max()));
1244   const char kInput[] = R"({
1245    'commandsFingerprint': '1',
1246    'stateFingerprint': '1',
1247    'traitsFingerprint': '1',
1248    'componentsFingerprint': '1',
1249    'waitTimeout': 10
1250   })";
1251   base::Closure callback;
1252   EXPECT_CALL(device_, PostDelayedTask(_, _, base::TimeDelta::FromSeconds(10)))
1253       .WillOnce(SaveArg<1>(&callback));
1254   EXPECT_JSON_EQ("{}", HandleRequest("/privet/v3/checkForUpdates", kInput));
1255   EXPECT_EQ(0, GetResponseCount());
1256   cloud_.NotifyOnTraitDefsChanged();
1257   EXPECT_EQ(1, GetResponseCount());
1258   const char kExpected[] = R"({
1259    'commandsFingerprint': '2',
1260    'stateFingerprint': '1',
1261    'traitsFingerprint': '2',
1262    'componentsFingerprint': '1'
1263   })";
1264   EXPECT_JSON_EQ(kExpected, GetResponse());
1265   callback.Run();
1266   EXPECT_EQ(1, GetResponseCount());
1267 }
1268 
1269 }  // namespace privet
1270 }  // namespace weave
1271