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.Collections.Generic;
\r
24 using System.Windows.Forms;
\r
26 namespace KancolleSniffer
\r
42 private LogType _logType;
\r
43 private readonly ShipMaster _shipMaster;
\r
44 private readonly ShipInfo _shipInfo;
\r
45 private readonly ItemInfo _itemInfo;
\r
46 private Action<string, string, string> _writer;
\r
47 private Func<DateTime> _nowFunc;
\r
48 public const string DateTimeFormat = @"yyyy\-MM\-dd HH\:mm\:ss";
\r
49 private dynamic _battle;
\r
50 private dynamic _map;
\r
51 private dynamic _basic;
\r
52 private int _kdockId;
\r
53 private DateTime _prevTime;
\r
54 private int _materialLogInterval = 10;
\r
55 private bool _start;
\r
57 public int MaterialLogInterval
\r
59 set { _materialLogInterval = value; }
\r
62 public string OutputDir
\r
64 set { _writer = new LogWriter(value).Write; }
\r
67 public Logger(ShipMaster master, ShipInfo ship, ItemInfo item)
\r
69 _shipMaster = master;
\r
72 _writer = new LogWriter().Write;
\r
73 _nowFunc = () => DateTime.Now;
\r
76 public void EnableLog(LogType type)
\r
81 public void SetWriter(Action<string, string, string> writer, Func<DateTime> nowFunc)
\r
87 public void InspectMissionResult(dynamic json)
\r
89 var r = (int)json.api_clear_result;
\r
90 var rstr = r == 2 ? "大成功" : r == 1 ? "成功" : "失敗";
\r
91 var material = new int[7];
\r
93 ((int[])json.api_get_material).CopyTo(material, 0);
\r
94 foreach (var i in new[] {1, 2})
\r
96 var attr = "api_get_item" + i;
\r
97 if (!json.IsDefined(attr) || json[attr].api_useitem_id != -1)
\r
99 var count = (int)json[attr].api_useitem_count;
\r
100 var flag = ((int[])json.api_useitem_flag)[i - 1];
\r
102 material[(int)Material.Bucket] = count;
\r
103 else if (flag == 2)
\r
104 material[(int)Material.Burner + 2] = count; // 高速建造材と開発資材が反対なのでいつか直す
\r
105 else if (flag == 3)
\r
106 material[(int)Material.Development - 2] = count;
\r
108 if ((_logType & LogType.Mission) != 0)
\r
111 string.Join(",", _nowFunc().ToString(DateTimeFormat),
\r
112 rstr, json.api_quest_name, string.Join(",", material)),
\r
113 "日付,結果,遠征,燃料,弾薬,鋼材,ボーキ,開発資材,高速修復材,高速建造材");
\r
117 public void InspectMapStart(dynamic json)
\r
123 public void InspectMapNext(dynamic json)
\r
128 public void InspectBattle(dynamic json)
\r
130 if (_battle != null) // 通常の夜戦は無視する
\r
135 public void InspectBattleResult(dynamic result)
\r
137 if ((_logType & LogType.Battle) == 0 || _map == null || _battle == null)
\r
139 _map = _battle = null;
\r
142 var fships = new List<string>();
\r
143 var deck = _shipInfo.GetDeck(_battle.api_dock_id() ? (int)_battle.api_dock_id - 1 : 0);
\r
144 fships.AddRange(deck.Select(id =>
\r
148 var s = _shipInfo[id];
\r
149 return string.Format("{0}(Lv{1}),{2}/{3}", s.Name, s.Level, s.NowHp, s.MaxHp);
\r
151 var edeck = ((int[])_battle.api_ship_ke).Skip(1).ToArray();
\r
152 var enowhp = ((int[])_battle.api_nowhps).Skip(7).ToArray();
\r
153 var emaxhp = ((int[])_battle.api_maxhps).Skip(7).ToArray();
\r
154 var eships = new List<string>();
\r
155 for (var i = 0; i < edeck.Count(); i++)
\r
157 eships.Add(edeck[i] == -1
\r
159 : string.Format("{0},{1}/{2}", _shipMaster[edeck[i]].Name, enowhp[i], emaxhp[i]));
\r
161 var cell = (int)_map.api_no;
\r
165 if (cell == (int)_map.api_bosscell_no || (int)_map.api_event_id == 5)
\r
166 boss = _start ? "出撃&ボス" : "ボス";
\r
167 _writer("海戦・ドロップ報告書", string.Join(",", _nowFunc().ToString(DateTimeFormat),
\r
168 result.api_quest_name,
\r
170 result.api_win_rank,
\r
171 BattleFormationName((int)_battle.api_formation[2]),
\r
172 FormationName(_battle.api_formation[0]),
\r
173 FormationName(_battle.api_formation[1]),
\r
174 result.api_enemy_info.api_deck_name,
\r
175 result.api_get_ship() ? result.api_get_ship.api_ship_type : "",
\r
176 result.api_get_ship() ? result.api_get_ship.api_ship_name : "",
\r
177 string.Join(",", fships),
\r
178 string.Join(",", eships)),
\r
179 "日付,海域,マス,ボス,ランク,艦隊行動,味方陣形,敵陣形,敵艦隊,ドロップ艦種,ドロップ艦娘," +
\r
180 "味方艦1,味方艦1HP,味方艦2,味方艦2HP,味方艦3,味方艦3HP,味方艦4,味方艦4HP,味方艦5,味方艦5HP,味方艦6,味方艦6HP," +
\r
181 "敵艦1,敵艦1HP,敵艦2,敵艦2HP,敵艦3,敵艦3HP,敵艦4,敵艦4HP,敵艦5,敵艦5HP,敵艦6,敵艦6HP"
\r
183 _map = _battle = null;
\r
187 private string FormationName(dynamic f)
\r
189 if (f is string) // 連合艦隊のときは文字列
\r
216 private static String BattleFormationName(int f)
\r
233 public void InspectBasic(dynamic json)
\r
238 public void InspectCreateItem(string request, dynamic json)
\r
240 if ((_logType & LogType.CreateItem) == 0)
\r
242 var values = HttpUtility.ParseQueryString(request);
\r
245 if (json.api_slot_item())
\r
247 var spec = _itemInfo.GetSpecByItemId((int)json.api_slot_item.api_slotitem_id);
\r
249 type = spec.TypeName;
\r
252 _nowFunc().ToString(DateTimeFormat) + "," +
\r
253 string.Join(",", name, type,
\r
254 values["api_item1"], values["api_item2"], values["api_item3"], values["api_item4"],
\r
255 Secretary(), _basic.api_level),
\r
256 "日付,開発装備,種別,燃料,弾薬,鋼材,ボーキ,秘書艦,司令部Lv");
\r
259 public void InspectCreateShip(string request)
\r
261 var values = HttpUtility.ParseQueryString(request);
\r
262 _kdockId = int.Parse(values["api_kdock_id"]);
\r
265 public void InspectKDock(dynamic json)
\r
267 if ((_logType & LogType.CreateShip) == 0 || _basic == null || _kdockId == 0)
\r
269 var kdock = ((dynamic[])json).First(e => e.api_id == _kdockId);
\r
270 var material = Enumerable.Range(1, 5).Select(i => (int)kdock["api_item" + i]).ToArray();
\r
271 var ship = _shipMaster[(int)kdock.api_created_ship_id];
\r
272 var avail = ((dynamic[])json).Count(e => (int)e.api_state == 0);
\r
274 _nowFunc().ToString(DateTimeFormat) + "," +
\r
275 string.Join(",", material.First() >= 1500 ? "大型艦建造" : "通常艦建造",
\r
276 ship.Name, ship.ShipTypeName, string.Join(",", material), avail, Secretary(), _basic.api_level),
\r
277 "日付,種類,名前,艦種,燃料,弾薬,鋼材,ボーキ,開発資材,空きドック,秘書艦,司令部Lv");
\r
281 private string Secretary()
\r
283 var ship = _shipInfo.GetShipStatuses(0)[0];
\r
284 return ship.Name + "(" + ship.Level + ")";
\r
287 public void InspectMaterial(dynamic json)
\r
289 if ((_logType & LogType.Material) == 0)
\r
291 var now = _nowFunc();
\r
292 if (now - _prevTime < TimeSpan.FromMinutes(_materialLogInterval))
\r
295 var material = new int[8];
\r
296 foreach (var e in json)
\r
297 material[(int)e.api_id - 1] = (int)e.api_value;
\r
299 now.ToString(DateTimeFormat) + "," +
\r
300 string.Join(",", material),
\r
301 "日付,燃料,弾薬,鋼材,ボーキ,高速建造材,高速修復材,開発資材,改修資材");
\r
305 public class LogWriter
\r
307 private readonly IFile _file;
\r
308 private readonly string _outputDir;
\r
310 public interface IFile
\r
312 string ReadAllText(string path);
\r
313 void AppendAllText(string path, string text);
\r
314 void Delete(string path);
\r
315 bool Exists(string path);
\r
318 private class FileWrapper : IFile
\r
320 // Shift_JISでないとExcelで文字化けする
\r
321 private readonly Encoding _encoding = Encoding.GetEncoding("Shift_JIS");
\r
323 public string ReadAllText(string path)
\r
325 return File.ReadAllText(path, _encoding);
\r
328 public void AppendAllText(string path, string text)
\r
330 File.AppendAllText(path, text, _encoding);
\r
333 public void Delete(string path)
\r
338 public bool Exists(string path)
\r
340 return File.Exists(path);
\r
344 public LogWriter(string outputDir = null, IFile file = null)
\r
346 _outputDir = outputDir ?? Path.GetDirectoryName(Application.ExecutablePath);
\r
347 _file = file ?? new FileWrapper();
\r
350 public void Write(string file, string s, string header)
\r
352 var path = Path.Combine(_outputDir, file);
\r
353 var csv = path + ".csv";
\r
354 var tmp = path + ".tmp";
\r
355 if (_file.Exists(tmp))
\r
359 _file.AppendAllText(csv, _file.ReadAllText(tmp));
\r
362 catch (IOException)
\r
366 if (!_file.Exists(csv))
\r
367 s = header + "\r\n" + s;
\r
368 foreach (var f in new[] {csv, tmp})
\r
372 _file.AppendAllText(f, s + "\r\n");
\r
375 catch (IOException)
\r