1 // Copyright (C) 2014, 2015 Kazuhiro Fujieda <fujieda@users.osdn.me>
\r
3 // Licensed under the Apache License, Version 2.0 (the "License");
\r
4 // you may not use this file except in compliance with the License.
\r
5 // You may obtain a copy of the License at
\r
7 // http://www.apache.org/licenses/LICENSE-2.0
\r
9 // Unless required by applicable law or agreed to in writing, software
\r
10 // distributed under the License is distributed on an "AS IS" BASIS,
\r
11 // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
\r
12 // See the License for the specific language governing permissions and
\r
13 // limitations under the License.
\r
16 using System.Collections.Generic;
\r
21 namespace KancolleSniffer
\r
38 private LogType _logType;
\r
39 private readonly ShipInfo _shipInfo;
\r
40 private readonly ItemInfo _itemInfo;
\r
41 private readonly BattleInfo _battleInfo;
\r
42 private Action<string, string, string> _writer;
\r
43 private Func<DateTime> _nowFunc;
\r
44 public const string DateTimeFormat = @"yyyy\-MM\-dd HH\:mm\:ss";
\r
45 private dynamic _battle;
\r
46 private dynamic _map;
\r
47 private dynamic _basic;
\r
48 private int _kdockId;
\r
49 private DateTime _prevTime;
\r
50 private int[] _currentMaterial = new int[Enum.GetValues(typeof(Material)).Length];
\r
51 private int _materialLogInterval = 10;
\r
52 private bool _start;
\r
54 public int MaterialLogInterval
\r
56 set { _materialLogInterval = value; }
\r
59 public string OutputDir
\r
61 set { _writer = new LogWriter(value).Write; }
\r
64 public Logger(ShipInfo ship, ItemInfo item, BattleInfo battle)
\r
68 _battleInfo = battle;
\r
69 _writer = new LogWriter().Write;
\r
70 _nowFunc = () => DateTime.Now;
\r
73 public void EnableLog(LogType type)
\r
78 public void SetWriter(Action<string, string, string> writer, Func<DateTime> nowFunc)
\r
84 public void InspectMissionResult(dynamic json)
\r
86 var r = (int)json.api_clear_result;
\r
87 var rstr = r == 2 ? "大成功" : r == 1 ? "成功" : "失敗";
\r
88 var material = new int[7];
\r
90 ((int[])json.api_get_material).CopyTo(material, 0);
\r
91 foreach (var i in new[] {1, 2})
\r
93 var attr = "api_get_item" + i;
\r
94 if (!json.IsDefined(attr) || json[attr].api_useitem_id != -1)
\r
96 var count = (int)json[attr].api_useitem_count;
\r
97 var flag = ((int[])json.api_useitem_flag)[i - 1];
\r
99 material[(int)Material.Bucket] = count;
\r
100 else if (flag == 2)
\r
101 material[(int)Material.Burner + 2] = count; // 高速建造材と開発資材が反対なのでいつか直す
\r
102 else if (flag == 3)
\r
103 material[(int)Material.Development - 2] = count;
\r
105 if ((_logType & LogType.Mission) != 0)
\r
108 string.Join(",", _nowFunc().ToString(DateTimeFormat),
\r
109 rstr, json.api_quest_name, string.Join(",", material)),
\r
110 "日付,結果,遠征,燃料,弾薬,鋼材,ボーキ,開発資材,高速修復材,高速建造材");
\r
114 public void InspectMapStart(dynamic json)
\r
119 if ((_logType & LogType.Material) != 0)
\r
120 WriteMaterialLog(_nowFunc());
\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 deckId = _battle.api_dock_id() ? (int)_battle.api_dock_id - 1 : 0;
\r
144 var deck = _shipInfo.GetDeck(deckId);
\r
145 fships.AddRange(deck.Select(id =>
\r
149 var s = _shipInfo.GetStatus(id);
\r
150 return $"{s.Name}(Lv{s.Level}),{s.NowHp}/{s.MaxHp}";
\r
152 var estatus = _battleInfo.EnemyResultStatus;
\r
153 var edeck = ((int[])_battle.api_ship_ke).Skip(1).ToArray();
\r
154 var eships = edeck.Select((id, i) =>
\r
158 var s = estatus[i];
\r
159 return $"{s.Name},{s.NowHp}/{s.MaxHp}";
\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 var dropType = result.api_get_ship() ? result.api_get_ship.api_ship_type : "";
\r
168 if (result.api_get_useitem())
\r
170 if (dropType == "")
\r
173 dropType += "+アイテム";
\r
175 var dropName = result.api_get_ship() ? result.api_get_ship.api_ship_name : "";
\r
176 if (result.api_get_useitem())
\r
178 var itemName = _itemInfo.GetUseItemName((int)result.api_get_useitem.api_useitem_id);
\r
179 if (dropName == "")
\r
180 dropName = itemName;
\r
182 dropName += "+" + itemName;
\r
184 var fp = _shipInfo.GetFighterPower(deckId);
\r
185 var fpower = fp[0] == fp[1] ? fp[0].ToString() : fp[0] + "~" + fp[1];
\r
186 _writer("海戦・ドロップ報告書", string.Join(",", _nowFunc().ToString(DateTimeFormat),
\r
187 result.api_quest_name,
\r
189 result.api_win_rank,
\r
190 BattleFormationName((int)_battle.api_formation[2]),
\r
191 FormationName(_battle.api_formation[0]),
\r
192 FormationName(_battle.api_formation[1]),
\r
193 result.api_enemy_info.api_deck_name,
\r
194 dropType, dropName,
\r
195 string.Join(",", fships),
\r
196 string.Join(",", eships),
\r
197 fpower, _battleInfo.EnemyFighterPower,
\r
198 AirControlLevelName(_battle)),
\r
199 "日付,海域,マス,ボス,ランク,艦隊行動,味方陣形,敵陣形,敵艦隊,ドロップ艦種,ドロップ艦娘," +
\r
200 "味方艦1,味方艦1HP,味方艦2,味方艦2HP,味方艦3,味方艦3HP,味方艦4,味方艦4HP,味方艦5,味方艦5HP,味方艦6,味方艦6HP," +
\r
201 "敵艦1,敵艦1HP,敵艦2,敵艦2HP,敵艦3,敵艦3HP,敵艦4,敵艦4HP,敵艦5,敵艦5HP,敵艦6,敵艦6HP," +
\r
204 _map = _battle = null;
\r
208 private string FormationName(dynamic f)
\r
210 if (f is string) // 連合艦隊のときは文字列
\r
237 private static string BattleFormationName(int f)
\r
254 private string AirControlLevelName(dynamic json)
\r
256 // BattleInfo.AirControlLevelは夜戦で消されているかもしれないので、こちらで改めて調べる。
\r
257 if (!json.api_kouku())
\r
259 var stage1 = json.api_kouku.api_stage1;
\r
260 if (stage1 == null)
\r
262 if (stage1.api_f_count == 0 && stage1.api_e_count == 0)
\r
264 switch ((int)stage1.api_disp_seiku)
\r
281 public void InspectBasic(dynamic json)
\r
286 public void InspectCreateItem(string request, dynamic json)
\r
288 if ((_logType & LogType.CreateItem) == 0)
\r
290 var values = HttpUtility.ParseQueryString(request);
\r
293 if (json.api_slot_item())
\r
295 var spec = _itemInfo.GetSpecByItemId((int)json.api_slot_item.api_slotitem_id);
\r
297 type = spec.TypeName;
\r
300 _nowFunc().ToString(DateTimeFormat) + "," +
\r
301 string.Join(",", name, type,
\r
302 values["api_item1"], values["api_item2"], values["api_item3"], values["api_item4"],
\r
303 Secretary(), _basic.api_level),
\r
304 "日付,開発装備,種別,燃料,弾薬,鋼材,ボーキ,秘書艦,司令部Lv");
\r
307 public void InspectCreateShip(string request)
\r
309 var values = HttpUtility.ParseQueryString(request);
\r
310 _kdockId = int.Parse(values["api_kdock_id"]);
\r
313 public void InspectKDock(dynamic json)
\r
315 if ((_logType & LogType.CreateShip) == 0 || _basic == null || _kdockId == 0)
\r
317 var kdock = ((dynamic[])json).First(e => e.api_id == _kdockId);
\r
318 var material = Enumerable.Range(1, 5).Select(i => (int)kdock["api_item" + i]).ToArray();
\r
319 var ship = _shipInfo.GetSpec((int)kdock.api_created_ship_id);
\r
320 var avail = ((dynamic[])json).Count(e => (int)e.api_state == 0);
\r
322 _nowFunc().ToString(DateTimeFormat) + "," +
\r
323 string.Join(",", material.First() >= 1500 ? "大型艦建造" : "通常艦建造",
\r
324 ship.Name, ship.ShipTypeName, string.Join(",", material), avail, Secretary(), _basic.api_level),
\r
325 "日付,種類,名前,艦種,燃料,弾薬,鋼材,ボーキ,開発資材,空きドック,秘書艦,司令部Lv");
\r
329 private string Secretary()
\r
331 var ship = _shipInfo.GetShipStatuses(0)[0];
\r
332 return ship.Name + "(" + ship.Level + ")";
\r
335 public void InspectMaterial(dynamic json)
\r
337 if ((_logType & LogType.Material) == 0)
\r
339 foreach (var e in json)
\r
340 _currentMaterial[(int)e.api_id - 1] = (int)e.api_value;
\r
341 var now = _nowFunc();
\r
342 if (now - _prevTime < TimeSpan.FromMinutes(_materialLogInterval))
\r
344 WriteMaterialLog(now);
\r
347 public void WriteMaterialLog(DateTime now)
\r
351 now.ToString(DateTimeFormat) + "," +
\r
352 string.Join(",", _currentMaterial),
\r
353 "日付,燃料,弾薬,鋼材,ボーキ,高速建造材,高速修復材,開発資材,改修資材");
\r
356 public void SetCurrentMaterial(int[] material)
\r
358 _currentMaterial = material;
\r
361 public void InspectRemodelSlot(string request, dynamic json)
\r
363 if ((_logType & LogType.RemodelSlot) == 0)
\r
365 var now = _nowFunc();
\r
366 var values = HttpUtility.ParseQueryString(request);
\r
367 var id = int.Parse(values["api_slot_id"]);
\r
368 var name = _itemInfo.GetName(id);
\r
369 var level = _itemInfo.GetStatus(id).Level;
\r
370 var success = (int)json.api_remodel_flag == 1 ? "○" : "×";
\r
371 var certain = int.Parse(values["api_certain_flag"]) == 1 ? "○" : "";
\r
374 if (json.api_use_slot_id())
\r
376 var use = (int[])json.api_use_slot_id;
\r
377 useName = _itemInfo.GetName(use[0]);
\r
378 useNum = use.Length.ToString();
\r
380 var after = (int[])json.api_after_material;
\r
381 var diff = new int[after.Length];
\r
382 for (var i = 0; i < after.Length; i++)
\r
383 diff[i] = _currentMaterial[i] - after[i];
\r
384 var ship1 = Secretary();
\r
386 var ships = _shipInfo.GetShipStatuses(0);
\r
387 if (ships.Length >= 2)
\r
388 ship2 = ships[1].Name + "(" + ships[1].Level + ")";
\r
390 now.ToString(DateTimeFormat) + "," +
\r
391 string.Join(",", name, level, success, certain, useName, useNum,
\r
392 diff[(int)Material.Fuel], diff[(int)Material.Bullet], diff[(int)Material.Steal],
\r
393 diff[(int)Material.Bouxite],
\r
394 diff[(int)Material.Development], diff[(int)Material.Screw],
\r
396 "日付,改修装備,レベル,成功,確実化,消費装備,消費数,燃料,弾薬,鋼材,ボーキ,開発資材,改修資材,秘書艦,二番艦");
\r
400 public class LogWriter
\r
402 private readonly IFile _file;
\r
403 private readonly string _outputDir;
\r
405 public interface IFile
\r
407 string ReadAllText(string path);
\r
408 void AppendAllText(string path, string text);
\r
409 void Delete(string path);
\r
410 bool Exists(string path);
\r
413 private class FileWrapper : IFile
\r
415 // Shift_JISでないとExcelで文字化けする
\r
416 private readonly Encoding _encoding = Encoding.GetEncoding("Shift_JIS");
\r
418 public string ReadAllText(string path) => File.ReadAllText(path, _encoding);
\r
420 public void AppendAllText(string path, string text)
\r
422 File.AppendAllText(path, text, _encoding);
\r
425 public void Delete(string path)
\r
430 public bool Exists(string path) => File.Exists(path);
\r
433 public LogWriter(string outputDir = null, IFile file = null)
\r
435 _outputDir = outputDir ?? AppDomain.CurrentDomain.BaseDirectory;
\r
436 _file = file ?? new FileWrapper();
\r
439 public void Write(string file, string s, string header)
\r
441 var path = Path.Combine(_outputDir, file);
\r
442 var csv = path + ".csv";
\r
443 var tmp = path + ".tmp";
\r
444 if (_file.Exists(tmp))
\r
448 _file.AppendAllText(csv, _file.ReadAllText(tmp));
\r
451 catch (IOException)
\r
455 if (!_file.Exists(csv))
\r
456 s = header + "\r\n" + s;
\r
457 foreach (var f in new[] {csv, tmp})
\r
461 _file.AppendAllText(f, s + "\r\n");
\r
464 catch (IOException e)
\r
467 throw new LogIOException("報告書の出力中にエラーが発生しました。", e);
\r
473 public class LogIOException : Exception
\r
475 public LogIOException()
\r
479 public LogIOException(string message) : base(message)
\r
483 public LogIOException(string message, Exception inner) : base(message, inner)
\r