OLD | NEW |
---|---|
(Empty) | |
1 // Copyright 2013 The Chromium 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 "remoting/host/pairing_registry_delegate_linux.h" | |
6 | |
7 #include "base/bind.h" | |
8 #include "base/environment.h" | |
9 #include "base/file_util.h" | |
10 #include "base/json/json_file_value_serializer.h" | |
11 #include "base/location.h" | |
12 #include "base/task_runner.h" | |
13 #include "base/values.h" | |
14 | |
15 namespace { | |
16 const char kConfigDir[] = ".config/chrome-remote-desktop"; | |
Sergey Ulanov
2013/05/31 18:45:46
There is remoting::GetConfigDir() that returns pat
Jamie
2013/06/19 17:56:44
Done.
| |
17 const char kRegistryFilename[] = "paired-clients"; | |
Sergey Ulanov
2013/05/31 18:45:46
Maybe paired-clients.json?
For host config we add
Jamie
2013/06/19 17:56:44
Done.
| |
18 const char kCreatedTimeLowKey[] = "created-time-low"; | |
19 const char kCreatedTimeHighKey[] = "created-time-high"; | |
20 const char kClientIdKey[] = "client-id"; | |
21 const char kClientNameKey[] = "client-name"; | |
22 const char kSharedSecretKey[] = "shared-secret"; | |
23 } // namespace | |
24 | |
25 | |
26 namespace remoting { | |
27 namespace protocol { | |
28 | |
29 PairingRegistryDelegateLinux::PairingRegistryDelegateLinux( | |
30 scoped_refptr<base::TaskRunner> task_runner) | |
31 : task_runner_(task_runner), | |
32 weak_factory_(this) { | |
33 } | |
34 | |
35 void PairingRegistryDelegateLinux::Save( | |
36 const PairingRegistry::PairedClients& paired_clients) { | |
37 task_runner_->PostTask( | |
38 FROM_HERE, | |
39 base::Bind(&PairingRegistryDelegateLinux::SaveOnCurrentThread, | |
40 weak_factory_.GetWeakPtr(), paired_clients)); | |
41 } | |
42 | |
43 void PairingRegistryDelegateLinux::SaveOnCurrentThread( | |
Sergey Ulanov
2013/05/31 18:45:46
Why is it "OnCurrentThread"? Should it be somethin
Jamie
2013/06/19 17:56:44
The new interface has different functions, but I'v
| |
44 const PairingRegistry::PairedClients& paired_clients) { | |
45 base::FilePath registry_path; | |
46 if (!GetRegistryFilePath(®istry_path)) { | |
47 return; | |
48 } | |
49 | |
50 base::ListValue root; | |
51 for (PairingRegistry::PairedClients::const_iterator i = | |
Sergey Ulanov
2013/05/31 18:45:46
It looks to me that most of the code in this class
Lambros
2013/05/31 18:51:59
This serialization code looks like it would be the
Jamie
2013/06/19 17:56:44
I was concerned about permissions. For example, on
Jamie
2013/06/19 17:56:44
I don't think all platforms will use JSON. On Wind
Sergey Ulanov
2013/06/19 18:23:50
What are the benefits of using registry instead of
| |
52 paired_clients.begin(); i != paired_clients.end(); ++i) { | |
53 base::DictionaryValue* pairing = new base::DictionaryValue(); | |
54 int64 created_time = i->second.created_time.ToInternalValue(); | |
55 pairing->SetInteger(kCreatedTimeLowKey, created_time & 0xffffffff); | |
56 pairing->SetInteger(kCreatedTimeHighKey, created_time >> 32); | |
57 pairing->SetString(kClientNameKey, i->second.client_name); | |
58 pairing->SetString(kClientIdKey, i->second.client_id); | |
59 pairing->SetString(kSharedSecretKey, i->second.shared_secret); | |
60 root.Append(pairing); | |
61 } | |
62 | |
63 JSONFileValueSerializer registry(registry_path); | |
Sergey Ulanov
2013/05/31 18:45:46
It's better to use base::ImportantFileWriter - it
Jamie
2013/06/19 17:56:44
Done.
| |
64 registry.Serialize(root); | |
Lambros
2013/05/31 18:51:59
I don't think JSONFileValueSerializer creates the
Jamie
2013/06/19 17:56:44
Done.
| |
65 } | |
66 | |
67 PairingRegistry::PairedClients | |
68 PairingRegistryDelegateLinux::LoadOnCurrentThread() { | |
69 PairingRegistry::PairedClients result; | |
70 base::FilePath registry_path; | |
71 if (!GetRegistryFilePath(®istry_path)) { | |
72 return result; | |
73 } | |
74 | |
75 JSONFileValueSerializer registry(registry_path); | |
76 int error_code; | |
77 std::string error_message; | |
78 scoped_ptr<base::Value> root( | |
79 registry.Deserialize(&error_code, &error_message)); | |
80 if (!root) { | |
81 LOG(ERROR) << "Failed to load paired clients: " << error_message | |
82 << " (" << error_code << ")."; | |
83 return result; | |
84 } | |
85 | |
86 base::ListValue* root_list = NULL; | |
87 if (!root->GetAsList(&root_list)) { | |
88 LOG(ERROR) << "Failed to load paired clients: root node is not a list."; | |
89 return result; | |
90 } | |
91 | |
92 for (size_t i = 0; i < root_list->GetSize(); ++i) { | |
93 base::DictionaryValue* pairing = NULL; | |
94 std::string client_id, client_name, shared_secret; | |
95 int created_time_low, created_time_high; | |
96 if (root_list->GetDictionary(i, &pairing) && | |
97 pairing->GetInteger(kCreatedTimeLowKey, &created_time_low) && | |
98 pairing->GetInteger(kCreatedTimeHighKey, &created_time_high) && | |
99 pairing->GetString(kClientIdKey, &client_id) && | |
100 pairing->GetString(kClientNameKey, &client_name) && | |
101 pairing->GetString(kSharedSecretKey, &shared_secret)) { | |
102 PairingRegistry::Pairing pairing_struct; | |
103 pairing_struct.created_time = base::Time::FromInternalValue( | |
104 (static_cast<int64>(created_time_high) << 32) + created_time_low); | |
Sergey Ulanov
2013/05/31 18:45:46
I'm not sure this will always work correctly. Part
Lambros
2013/05/31 19:55:26
Ah yes, I missed this! I think created_time_low/hi
Jamie
2013/06/19 17:56:44
Done.
Jamie
2013/06/19 17:56:44
I've followed Lambros's suggestion of using string
| |
105 pairing_struct.client_id = client_id; | |
106 pairing_struct.client_name = client_name; | |
107 pairing_struct.shared_secret = shared_secret; | |
108 result[client_id] = pairing_struct; | |
109 } else { | |
110 LOG(ERROR) << "Paired client " << i << " has unexpected format."; | |
111 } | |
112 } | |
113 | |
114 return result; | |
115 } | |
116 | |
117 bool PairingRegistryDelegateLinux::GetRegistryFilePath(base::FilePath* path) { | |
118 if (!filename_for_testing_.empty()) { | |
119 *path = filename_for_testing_; | |
120 return true; | |
121 } | |
122 | |
123 base::FilePath home = file_util::GetHomeDir(); | |
124 if (home.empty()) { | |
125 LOG(ERROR) << "Could not locate user's home directory."; | |
126 return false; | |
127 } | |
128 | |
129 *path = home.Append(kConfigDir).Append(kRegistryFilename); | |
130 return true; | |
131 } | |
132 | |
133 void PairingRegistryDelegateLinux::SetFilenameForTesting( | |
134 const base::FilePath &filename) { | |
135 filename_for_testing_ = filename; | |
136 } | |
137 | |
138 | |
139 scoped_ptr<PairingRegistry::Delegate> PairingRegistry::CreateDelegate( | |
140 scoped_refptr<base::TaskRunner> task_runner) { | |
141 return scoped_ptr<PairingRegistry::Delegate>( | |
142 new PairingRegistryDelegateLinux(task_runner)); | |
143 } | |
144 | |
145 } // namespace protocol | |
146 } // namespace remoting | |
OLD | NEW |