1 // Copyright (C) 2014, 2015 Kazuhiro Fujieda <fujieda@users.osdn.me>
\r
3 // This program is part of KancolleSniffer.
\r
5 // KancolleSniffer is free software: you can redistribute it and/or modify
\r
6 // it under the terms of the GNU General Public License as published by
\r
7 // the Free Software Foundation, either version 3 of the License, or
\r
8 // (at your option) any later version.
\r
10 // This program is distributed in the hope that it will be useful,
\r
11 // but WITHOUT ANY WARRANTY; without even the implied warranty of
\r
12 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
\r
13 // GNU General Public License for more details.
\r
15 // You should have received a copy of the GNU General Public License
\r
16 // along with this program; if not, see <http://www.gnu.org/licenses/>.
\r
19 using System.Globalization;
\r
23 using System.Net.Sockets;
\r
25 using System.Threading;
\r
27 using System.Windows.Forms;
\r
29 namespace KancolleSniffer
\r
31 public class LogServer
\r
33 private readonly TcpListener _listener;
\r
34 private readonly string _indexDir = Path.GetDirectoryName(Application.ExecutablePath);
\r
35 private string _outputDir = Path.GetDirectoryName(Application.ExecutablePath);
\r
37 public int Port { get; private set; }
\r
39 public string OutputDir
\r
41 set { _outputDir = value; }
\r
44 public LogServer(int port)
\r
47 _listener = new TcpListener(IPAddress.Loopback, port);
\r
53 new Thread(Listen).Start();
\r
56 private void Listen()
\r
62 var socket = _listener.AcceptSocket();
\r
63 new Thread(Process).Start(socket);
\r
66 catch (SocketException)
\r
75 private void Process(Object obj)
\r
77 var client = (Socket)obj;
\r
78 var data = new byte[4096];
\r
79 var from = DateTime.MinValue;
\r
80 var to = DateTime.MaxValue;
\r
83 if (client.Receive(data) == 0)
\r
85 var request = Encoding.UTF8.GetString(data).Split('\r')[0].Split(' ');
\r
86 if (request.Length != 3)
\r
88 SendError(client, "400 Bad Request");
\r
91 if (!request[0].StartsWith("GET", StringComparison.OrdinalIgnoreCase))
\r
93 SendError(client, "501 Not Implemented");
\r
96 var tmp = request[1].Split('?');
\r
97 var path = HttpUtility.UrlDecode(tmp[0]);
\r
98 if (path == null || !path.StartsWith("/"))
\r
100 SendError(client, "400 Bad Request");
\r
103 if (tmp.Length == 2)
\r
105 var query = HttpUtility.ParseQueryString(tmp[1]);
\r
106 if (query["from"] != null)
\r
109 double.TryParse(query["from"], out tick);
\r
110 from = new DateTime(1970, 1, 1).ToLocalTime().AddSeconds(tick / 1000);
\r
112 if (query["to"] != null)
\r
115 double.TryParse(query["to"], out tick);
\r
116 to = new DateTime(1970, 1, 1).ToLocalTime().AddSeconds(tick / 1000);
\r
120 path = path == "/" ? "index.html" : path.Substring(1);
\r
121 var full = Path.Combine(_indexDir, path);
\r
122 var csv = Path.Combine(_outputDir, path);
\r
123 if (path.EndsWith(".html", StringComparison.OrdinalIgnoreCase) && File.Exists(full))
\r
125 SendFile(client, full, "text/html");
\r
128 if (path.EndsWith(".csv", StringComparison.OrdinalIgnoreCase) && File.Exists(csv))
\r
130 SendFile(client, csv, "text/csv; charset=Shift_JIS");
\r
133 if (path.EndsWith(".json", StringComparison.OrdinalIgnoreCase))
\r
135 SendJsonData(client, csv, from, to);
\r
138 if (path.EndsWith(".js", StringComparison.OrdinalIgnoreCase) && File.Exists(full))
\r
140 SendFile(client, full, "application/javascript");
\r
143 SendError(client, "404 Not Found");
\r
145 catch (IOException)
\r
148 catch (SocketException)
\r
157 private void SendError(Socket client, string error)
\r
159 using (var writer = new StreamWriter(new MemoryStream(), Encoding.ASCII))
\r
161 writer.Write("HTTP/1.1 {0}\r\n", error);
\r
162 writer.Write("Server: KancolleSniffer\r\n");
\r
163 writer.Write("Date: {0:R}\r\n", DateTime.Now);
\r
164 writer.Write("Connection: close\r\n\r\n");
\r
165 writer.Write("<html><head><title>{0}</title></head>\r\n", error);
\r
166 writer.Write("<body><h4>{0}</h4></body></html>\r\n\r\n", error);
\r
168 client.Send(((MemoryStream)writer.BaseStream).ToArray());
\r
172 private void SendJsonData(Socket client, string path, DateTime from, DateTime to)
\r
174 var header = new StreamWriter(new MemoryStream(), Encoding.ASCII);
\r
175 header.Write("HTTP/1.1 200 OK\r\n");
\r
176 header.Write("Server: KancolleSniffer\r\n");
\r
177 header.Write("Date: {0:R}\r\n", DateTime.Now);
\r
178 header.Write("Content-Type: {0}\r\n", "application/json; charset=Shift_JIS");
\r
179 header.Write("Connection: close\r\n\r\n");
\r
181 client.Send(((MemoryStream)header.BaseStream).ToArray());
\r
183 var csv = path.Replace(".json", ".csv");
\r
184 var encoding = Encoding.GetEncoding("Shift_JIS");
\r
185 client.Send(encoding.GetBytes("{ \"data\": [\n"));
\r
186 if (File.Exists(csv))
\r
188 var delimiter = "";
\r
189 var material = path.EndsWith("資材ログ.json"); // 末尾の空データを削除する必要がある
\r
190 foreach (var line in File.ReadLines(csv, encoding).Skip(1))
\r
192 var data = line.Split(',');
\r
194 if (!DateTime.TryParseExact(data[0], Logger.DateTimeFormat, CultureInfo.InvariantCulture,
\r
195 DateTimeStyles.AssumeLocal, out date) &&
\r
196 DateTime.TryParse(data[0], CultureInfo.CurrentCulture, DateTimeStyles.AssumeLocal, out date))
\r
198 data[0] = date.ToString(Logger.DateTimeFormat);
\r
200 if (date < from || to < date)
\r
202 client.Send(encoding.GetBytes(delimiter + "[\"" +
\r
203 string.Join("\",\"", (material ? data.Take(9) : data)) + "\"]"));
\r
207 client.Send(encoding.GetBytes("]}\n"));
\r
210 private void SendFile(Socket client, string path, string mime)
\r
212 using (var writer = new StreamWriter(new MemoryStream(), Encoding.ASCII))
\r
214 writer.Write("HTTP/1.1 200 OK\r\n");
\r
215 writer.Write("Server: KancolleSniffer\r\n");
\r
216 writer.Write("Date: {0:R}\r\n", DateTime.Now);
\r
217 writer.Write("Content-Length: {0}\r\n", new FileInfo(path).Length);
\r
218 writer.Write("Content-Type: {0}\r\n", mime);
\r
219 writer.Write("Connection: close\r\n\r\n");
\r
221 client.SendFile(path, ((MemoryStream)writer.BaseStream).ToArray(), null,
\r
222 TransmitFileOptions.UseDefaultWorkerThread);
\r
228 _listener.Server.Close();
\r