|
12 | 12 | */
|
13 | 13 | package com.qubole.rubix.prestosql;
|
14 | 14 |
|
15 |
| -import com.google.common.base.Throwables; |
16 |
| -import com.google.common.collect.ImmutableList; |
17 |
| -import com.google.common.collect.Lists; |
18 |
| -import com.google.common.reflect.TypeToken; |
19 |
| -import com.google.gson.Gson; |
20 |
| -import com.qubole.rubix.common.utils.ClusterUtil; |
21 |
| -import com.qubole.rubix.spi.ClusterManager; |
| 15 | +import com.qubole.rubix.spi.AsyncClusterManager; |
22 | 16 | import com.qubole.rubix.spi.ClusterType;
|
23 | 17 | import io.prestosql.spi.Node;
|
24 | 18 | import io.prestosql.spi.NodeManager;
|
25 | 19 | import org.apache.commons.logging.Log;
|
26 | 20 | import org.apache.commons.logging.LogFactory;
|
27 | 21 | import org.apache.hadoop.conf.Configuration;
|
28 | 22 |
|
29 |
| -import javax.annotation.Nullable; |
30 |
| - |
31 |
| -import java.io.BufferedReader; |
32 |
| -import java.io.IOException; |
33 |
| -import java.io.InputStreamReader; |
34 |
| -import java.lang.reflect.Type; |
35 |
| -import java.net.HttpURLConnection; |
36 |
| -import java.net.MalformedURLException; |
37 |
| -import java.net.URI; |
38 |
| -import java.net.URL; |
39 | 23 | import java.net.UnknownHostException;
|
40 |
| -import java.util.HashSet; |
41 |
| -import java.util.List; |
42 |
| -import java.util.Objects; |
43 | 24 | import java.util.Set;
|
44 |
| -import java.util.stream.Collectors; |
45 |
| - |
46 |
| -import static java.util.Objects.requireNonNull; |
47 | 25 |
|
48 | 26 | /**
|
49 | 27 | * Created by stagra on 14/1/16.
|
50 | 28 | */
|
51 |
| -public class PrestoClusterManager extends ClusterManager |
| 29 | +public class PrestoClusterManager extends AsyncClusterManager |
52 | 30 | {
|
53 |
| - private static final String DEFAULT_USER = "rubix"; |
54 |
| - private int serverPort = 8081; |
55 |
| - private String serverAddress = "localhost"; |
56 |
| - |
57 |
| - private Log log = LogFactory.getLog(PrestoClusterManager.class); |
58 |
| - |
59 |
| - @Nullable |
60 |
| - private static volatile NodeManager nodeManager; |
61 |
| - public static String serverPortConf = "caching.fs.presto-server-port"; |
62 |
| - |
63 |
| - // Safe to use single instance of HttpClient since Supplier.get() provides synchronization |
64 |
| - @Override |
65 |
| - public void initialize(Configuration conf) |
66 |
| - throws UnknownHostException |
67 |
| - { |
68 |
| - super.initialize(conf); |
69 |
| - this.serverPort = conf.getInt(serverPortConf, serverPort); |
70 |
| - this.serverAddress = ClusterUtil.getMasterHostname(conf); |
71 |
| - } |
72 |
| - |
73 |
| - @Override |
74 |
| - public List<String> getNodesInternal() |
75 |
| - { |
76 |
| - if (nodeManager != null) { |
77 |
| - return getNodesFromNodeManager(); |
78 |
| - } |
79 |
| - |
80 |
| - try { |
81 |
| - URL allNodesRequest = getNodeUrl(); |
82 |
| - URL failedNodesRequest = getFailedNodeUrl(); |
83 |
| - |
84 |
| - HttpURLConnection allHttpCon = getHttpURLConnection(allNodesRequest); |
85 |
| - |
86 |
| - int allNodesResponseCode = allHttpCon.getResponseCode(); |
87 |
| - |
88 |
| - StringBuilder allResponse = new StringBuilder(); |
89 |
| - StringBuilder failedResponse = new StringBuilder(); |
90 |
| - try { |
91 |
| - if (allNodesResponseCode == HttpURLConnection.HTTP_OK) { |
92 |
| - BufferedReader in = new BufferedReader(new InputStreamReader(allHttpCon.getInputStream())); |
93 |
| - String inputLine = ""; |
94 |
| - try { |
95 |
| - while ((inputLine = in.readLine()) != null) { |
96 |
| - allResponse.append(inputLine); |
97 |
| - } |
98 |
| - } |
99 |
| - catch (IOException e) { |
100 |
| - throw new IOException(e); |
101 |
| - } |
102 |
| - finally { |
103 |
| - in.close(); |
104 |
| - } |
105 |
| - } |
106 |
| - else { |
107 |
| - log.warn("v1/node failed with code: " + allNodesResponseCode); |
108 |
| - return null; |
109 |
| - } |
110 |
| - } |
111 |
| - catch (IOException e) { |
112 |
| - throw new IOException(e); |
113 |
| - } |
114 |
| - finally { |
115 |
| - allHttpCon.disconnect(); |
116 |
| - } |
117 |
| - |
118 |
| - HttpURLConnection failHttpConn = getHttpURLConnection(failedNodesRequest); |
119 |
| - int failedNodesResponseCode = failHttpConn.getResponseCode(); |
120 |
| - // check on failed nodes |
121 |
| - try { |
122 |
| - if (failedNodesResponseCode == HttpURLConnection.HTTP_OK) { |
123 |
| - BufferedReader in = new BufferedReader(new InputStreamReader(failHttpConn.getInputStream())); |
124 |
| - String inputLine; |
125 |
| - try { |
126 |
| - while ((inputLine = in.readLine()) != null) { |
127 |
| - failedResponse.append(inputLine); |
128 |
| - } |
129 |
| - } |
130 |
| - catch (IOException e) { |
131 |
| - throw new IOException(e); |
132 |
| - } |
133 |
| - finally { |
134 |
| - in.close(); |
135 |
| - } |
136 |
| - } |
137 |
| - } |
138 |
| - catch (IOException e) { |
139 |
| - throw new IOException(e); |
140 |
| - } |
141 |
| - finally { |
142 |
| - failHttpConn.disconnect(); |
143 |
| - } |
| 31 | + private static Log log = LogFactory.getLog(PrestoClusterManager.class); |
| 32 | + static volatile NodeManager NODE_MANAGER; |
144 | 33 |
|
145 |
| - Gson gson = new Gson(); |
146 |
| - Type type = new TypeToken<List<Stats>>() |
147 |
| - { |
148 |
| - }.getType(); |
149 |
| - |
150 |
| - List<Stats> allNodes = gson.fromJson(allResponse.toString(), type); |
151 |
| - List<Stats> failedNodes = gson.fromJson(failedResponse.toString(), type); |
152 |
| - |
153 |
| - if (failedNodes.isEmpty()) { |
154 |
| - failedNodes = ImmutableList.of(); |
155 |
| - } |
156 |
| - |
157 |
| - // keep only the healthy nodes |
158 |
| - allNodes.removeAll(failedNodes); |
159 |
| - |
160 |
| - Set<String> hosts = new HashSet<String>(); |
161 |
| - for (Stats node : allNodes) { |
162 |
| - hosts.add(node.getUri().getHost()); |
163 |
| - } |
164 |
| - |
165 |
| - return Lists.newArrayList(hosts.toArray(new String[0])); |
166 |
| - } |
167 |
| - catch (IOException e) { |
168 |
| - throw Throwables.propagate(e); |
169 |
| - } |
170 |
| - } |
171 |
| - |
172 |
| - private HttpURLConnection getHttpURLConnection(URL urlRequest) |
173 |
| - throws IOException |
| 34 | + public static void setNodeManager(NodeManager nodeManager) |
174 | 35 | {
|
175 |
| - requireNonNull(urlRequest, "urlRequest is null"); |
176 |
| - HttpURLConnection allHttpCon = (HttpURLConnection) urlRequest.openConnection(); |
177 |
| - allHttpCon.setConnectTimeout(500); //ms |
178 |
| - allHttpCon.setRequestMethod("GET"); |
179 |
| - allHttpCon.setRequestProperty("X-Presto-User", DEFAULT_USER); |
180 |
| - return allHttpCon; |
| 36 | + PrestoClusterManager.NODE_MANAGER = nodeManager; |
181 | 37 | }
|
182 | 38 |
|
183 |
| - private List<String> getNodesFromNodeManager() |
184 |
| - { |
185 |
| - requireNonNull(nodeManager, "nodeManager is null"); |
186 |
| - List<String> workers = nodeManager.getWorkerNodes().stream() |
187 |
| - .filter(node -> !node.isCoordinator()) |
188 |
| - .map(Node::getHost) |
189 |
| - .collect(Collectors.toList()); |
190 |
| - |
191 |
| - return workers; |
192 |
| - } |
| 39 | + private volatile NodeManager nodeManager; |
193 | 40 |
|
194 | 41 | @Override
|
195 |
| - protected String getCurrentNodeHostname() |
| 42 | + public void initialize(Configuration conf) throws UnknownHostException |
196 | 43 | {
|
197 |
| - if (nodeManager != null) { |
198 |
| - return nodeManager.getCurrentNode().getHost(); |
| 44 | + super.initialize(conf); |
| 45 | + nodeManager = NODE_MANAGER; |
| 46 | + if (nodeManager == null) { |
| 47 | + nodeManager = new StandaloneNodeManager(conf); |
199 | 48 | }
|
200 |
| - |
201 |
| - return super.getCurrentNodeHostname(); |
202 | 49 | }
|
203 | 50 |
|
204 | 51 | @Override
|
205 |
| - protected String getCurrentNodeHostAddress() |
| 52 | + public Set<String> getNodesInternal() |
206 | 53 | {
|
207 |
| - if (nodeManager != null) { |
208 |
| - try { |
209 |
| - return nodeManager.getCurrentNode().getHostAndPort().toInetAddress().getHostAddress(); |
210 |
| - } |
211 |
| - catch (UnknownHostException e) { |
212 |
| - log.warn("Could not get HostAddress from NodeManager", e); |
213 |
| - // fallback |
214 |
| - } |
215 |
| - } |
216 |
| - |
217 |
| - return super.getCurrentNodeHostAddress(); |
| 54 | + return ClusterManagerNodeGetter.getNodesInternal(nodeManager); |
218 | 55 | }
|
219 | 56 |
|
220 | 57 | @Override
|
221 | 58 | public ClusterType getClusterType()
|
222 | 59 | {
|
223 | 60 | return ClusterType.PRESTOSQL_CLUSTER_MANAGER;
|
224 | 61 | }
|
225 |
| - |
226 |
| - public static void setPrestoServerPort(Configuration conf, int port) |
227 |
| - { |
228 |
| - conf.setInt(serverPortConf, port); |
229 |
| - } |
230 |
| - |
231 |
| - public static void setNodeManager(NodeManager nodeManager) |
232 |
| - { |
233 |
| - PrestoClusterManager.nodeManager = requireNonNull(nodeManager, "nodeManager is null"); |
234 |
| - } |
235 |
| - |
236 |
| - private URL getNodeUrl() |
237 |
| - throws MalformedURLException |
238 |
| - { |
239 |
| - return new URL("http://" + serverAddress + ":" + serverPort + "/v1/node"); |
240 |
| - } |
241 |
| - |
242 |
| - private URL getFailedNodeUrl() |
243 |
| - throws MalformedURLException |
244 |
| - { |
245 |
| - return new URL("http://" + serverAddress + ":" + serverPort + "/v1/node/failed"); |
246 |
| - } |
247 |
| - |
248 |
| - public static class Stats |
249 |
| - { |
250 |
| - URI uri; |
251 |
| - String lastResponseTime; |
252 |
| - |
253 |
| - public Stats() |
254 |
| - { |
255 |
| - } |
256 |
| - |
257 |
| - public Stats(URI uri, String lastResponseTime) |
258 |
| - { |
259 |
| - this.uri = uri; |
260 |
| - this.lastResponseTime = lastResponseTime; |
261 |
| - } |
262 |
| - |
263 |
| - public URI getUri() |
264 |
| - { |
265 |
| - return uri; |
266 |
| - } |
267 |
| - |
268 |
| - public void setURI(URI uri) |
269 |
| - { |
270 |
| - this.uri = uri; |
271 |
| - } |
272 |
| - |
273 |
| - String getLastResponseTime() |
274 |
| - { |
275 |
| - return lastResponseTime; |
276 |
| - } |
277 |
| - |
278 |
| - public void setLastResponseTime(String lastResponseTime) |
279 |
| - { |
280 |
| - this.lastResponseTime = lastResponseTime; |
281 |
| - } |
282 |
| - |
283 |
| - @Override |
284 |
| - public boolean equals(Object other) |
285 |
| - { |
286 |
| - if (this == other) { |
287 |
| - return true; |
288 |
| - } |
289 |
| - if (other == null || getClass() != other.getClass()) { |
290 |
| - return false; |
291 |
| - } |
292 |
| - Stats o = (Stats) other; |
293 |
| - |
294 |
| - if (!uri.equals(o.getUri())) { |
295 |
| - return false; |
296 |
| - } |
297 |
| - |
298 |
| - if (lastResponseTime != null && o.getLastResponseTime() != null) { |
299 |
| - return lastResponseTime.equals(o.getLastResponseTime()); |
300 |
| - } |
301 |
| - |
302 |
| - return lastResponseTime == o.getLastResponseTime(); |
303 |
| - } |
304 |
| - |
305 |
| - @Override |
306 |
| - public int hashCode() |
307 |
| - { |
308 |
| - return Objects.hash(uri, lastResponseTime); |
309 |
| - } |
310 |
| - } |
311 | 62 | }
|
0 commit comments