ZoneService.cs 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734
  1. using CommonAI;
  2. using CommonAI.Zone.Instance;
  3. using CommonAI.ZoneServer.JSGModule;
  4. using CommonLang;
  5. using CommonLang.Concurrent;
  6. using CommonLang.Log;
  7. using Pomelo;
  8. using System;
  9. using System.Collections.Generic;
  10. using System.Configuration;
  11. using System.Diagnostics;
  12. using System.Dynamic;
  13. using System.Threading;
  14. using System.Web.Helpers;
  15. using XmdsCommon.Plugin;
  16. using XmdsCommonServer.Plugin;
  17. using XmdsServerNode.Node;
  18. using XmdsServerNode.Node.R2bNotify;
  19. using static CommonAI.ZoneServer.JSGModule.JSGServerProfile;
  20. namespace XmdsServerEdgeJS.Zone
  21. {
  22. public abstract class ZoneService : IZone
  23. {
  24. //战斗服版本号
  25. private static readonly string S_BSVersion = "1.0.1";
  26. public static ZoneService Instance { get; private set; }
  27. protected ZoneService()
  28. {
  29. Instance = this;
  30. updatePrivateMemory(null);
  31. }
  32. //----------------------------------------------------------------------------------------------
  33. protected Logger log = LoggerFactory.GetLogger(typeof(ZoneService).Name);
  34. protected Logger monitorLog = LoggerFactory.GetLogger("monitor");
  35. private ZoneNodeManager zoneNodeManager;
  36. private BattleCodec codec;
  37. private AtomicLong privateMemoryMB = new AtomicLong(0);
  38. private Timer systemUpdateTimer;
  39. private Timer logUpdateTimer;
  40. protected XmdsServerProxy b2r_proxy = new XmdsServerProxy();
  41. /// <summary>
  42. /// 副本列表
  43. /// </summary>
  44. private HashMap<string, XmdsZoneNode> nodes = new HashMap<string, XmdsZoneNode>();
  45. /// <summary>
  46. /// 玩家列表
  47. /// </summary>
  48. private HashMap<string, XmdsPlayer> players = new HashMap<string, XmdsPlayer>();
  49. //----------------------------------------------------------------------------------------------
  50. public int PlayerCount
  51. {
  52. get
  53. {
  54. lock (players)
  55. {
  56. return players.Count;
  57. }
  58. }
  59. }
  60. public int ZoneNodeCount
  61. {
  62. get
  63. {
  64. lock (nodes)
  65. {
  66. return nodes.Count;
  67. }
  68. }
  69. }
  70. public long ProcessPrivateMemoryMB
  71. {
  72. get { return privateMemoryMB.Value; }
  73. }
  74. //----------------------------------------------------------------------------------------------
  75. #region Internal
  76. public XmdsPlayer getPlayer(string playerID)
  77. {
  78. lock (players)
  79. {
  80. return players.Get(playerID);
  81. }
  82. }
  83. protected XmdsZoneNode getZoneNode(string instanceID)
  84. {
  85. lock (nodes)
  86. {
  87. return nodes.Get(instanceID);//[instanceID];
  88. }
  89. }
  90. //internal void sendToGameServer(string name, Object param)
  91. //{
  92. // try
  93. // {
  94. // //转化为json
  95. // string json = Json.Encode(param);
  96. // IceManager.instance().eventNotify(name, json);
  97. // }
  98. // catch (Exception err)
  99. // {
  100. // log.Error(err.Message, err);
  101. // }
  102. //}
  103. protected void notifyBattleServer(string instanceId, R2BNotifyMessage param)
  104. {
  105. var node = getZoneNode(instanceId);
  106. if (node != null)
  107. {
  108. if(param is TriggerEventR2B triggerO)
  109. {
  110. node.Node.Zone.RecvMessageFromGameServer(new CommonAI.ZoneServer.SendMessageR2B() { Message = triggerO.message, TriggerUnits = triggerO.TriggerUnits });
  111. }
  112. else
  113. {
  114. //b2r_proxy.SendToBattleServer(node.Node.Zone, param);
  115. param.OnHandle(node.Node.Zone);
  116. }
  117. }
  118. }
  119. private void updatePrivateMemory(object state)
  120. {
  121. long private_memory_mb = Process.GetCurrentProcess().PrivateMemorySize64 / 1024 / 1024;
  122. privateMemoryMB.Value = private_memory_mb;
  123. }
  124. private void updateLog(object state)
  125. {
  126. dynamic eventStat = new ExpandoObject();
  127. eventStat.Type = "monitor";
  128. eventStat.Object = InstanceZoneObject.ActiveObjectCount + "/" + InstanceZoneObject.AllocObjectCount;
  129. eventStat.Zone = InstanceZone.ActiveZoneCount + "/" + InstanceZone.AllocZoneCount;
  130. eventStat.Node = ZoneNodeCount;
  131. eventStat.Player = PlayerCount;
  132. eventStat.Memory_MB = ProcessPrivateMemoryMB;
  133. eventStat.Pool = ObjectPoolStatus.TotalActive + "/" + ObjectPoolStatus.TotalCount;
  134. monitorLog.Debug(eventStat);
  135. // 输出统计日志
  136. if (JSGServerProfile.CheckPrintAllData())
  137. {
  138. this.PrintAllSceneInfo();
  139. }
  140. }
  141. private void PrintAllSceneInfo()
  142. {
  143. lock (nodes)
  144. {
  145. foreach (var p in new List<XmdsZoneNode>(nodes.Values))
  146. {
  147. try
  148. {
  149. string ext = p.Node.Zone.IsObjectMapNull() ? ", 场景已销毁:" : ", units=" + p.Node.Zone.AllUnitsCount
  150. + ", spells=" + p.Node.Zone.AllSpellsCount + ", items=" + p.Node.Zone.AllItemsCount + ", players: " + p.Node.Zone.AllPlayersCount;
  151. log.Info("PrintAllSceneInfo-ID:" + p.InstanceID + ", " + p.Node.GetBindGameSrvId() + ", 场景ID: " + p.Node.SceneID + ext);
  152. }
  153. catch (Exception e)
  154. {
  155. log.Error("PrintAllSceneInfo: " + p.InstanceID + ", e: " + e);
  156. }
  157. }
  158. }
  159. }
  160. #endregion
  161. //----------------------------------------------------------------------------------------------
  162. public void Start(ZoneConfig zoneConfig)
  163. {
  164. lock (this)
  165. {
  166. try
  167. {
  168. log.Info("startPath:" + zoneConfig.startPath);
  169. log.Info("assetPath:" + zoneConfig.assetPath);
  170. JSGServerProfile.init();
  171. systemUpdateTimer = new Timer(updatePrivateMemory, this, 0, 10000);
  172. zoneNodeManager = new ZoneNodeManager();
  173. zoneNodeManager.Init(zoneConfig.startPath, b2r_proxy, zoneConfig.assetPath);
  174. codec = new BattleCodec(ZoneNodeManager.Templates.Templates);
  175. //日志定时器
  176. logUpdateTimer = new Timer(updateLog, this, 120000, 120000);
  177. }
  178. catch (Exception err)
  179. {
  180. //log.Error("初始化失败");
  181. //log.Error(err.Message, err);
  182. //Environment.Exit(0);
  183. throw err;
  184. }
  185. }
  186. }
  187. /** 网络重置 */
  188. public void onNetReconnect(string connetorId, IFastSession session)
  189. {
  190. try
  191. {
  192. lock (players)
  193. {
  194. foreach (var player in players.Values)
  195. {
  196. player.OnNetReconnect(connetorId, session);
  197. }
  198. }
  199. }
  200. catch(Exception e)
  201. {
  202. log.Error("onNetReconnect catch: " + connetorId + ", " + e);
  203. }
  204. }
  205. public void Stop()
  206. {
  207. this.ClearAllPlayers((e, c) => { });
  208. this.DestoryAllZones((e, c) => { });
  209. lock (this)
  210. {
  211. if (zoneNodeManager != null)
  212. {
  213. b2r_proxy.Dispose();
  214. zoneNodeManager.Shutdown();
  215. systemUpdateTimer.Dispose();
  216. logUpdateTimer.Dispose();
  217. zoneNodeManager = null;
  218. codec = null;
  219. }
  220. }
  221. }
  222. public void SetCallBack(string gameSrvId)
  223. {
  224. //JSGMountainKingModule.Init(gameSrvId);
  225. }
  226. public void CreateZone(string playerId, string gameServerId, int mapTemplateId, string instanceId, bool forceCreate,
  227. string data, Action<int> cb, Action<Exception> err)
  228. {
  229. long lstart = TimeUtil.GetTimestampMS();
  230. int resCode = 0;
  231. if (!forceCreate && playerId != null && playerId.Length > 0)
  232. {
  233. var player = getPlayer(playerId);
  234. if (player != null && player.BindingActor.IsActive && player.BindingActor.Virtual.IsInPVP())
  235. {
  236. resCode = 1;
  237. XmdsVirtual playerVirtual = player.BindingActor.Virtual as XmdsVirtual;
  238. log.Warn("PVP状态下传送2:" + playerVirtual.mUnit.PlayerUUID + ", 场景ID: " + playerVirtual.mUnit.Parent.GetSceneID() + ",hateInfo: "
  239. + playerVirtual.GetHateSystem().GetHatePlayerInfo() + ", 触发PVP玩家:" + playerVirtual.mPvpTriggerPlayerId);
  240. }
  241. }
  242. if (resCode == 0)
  243. {
  244. long private_memory_mb = ProcessPrivateMemoryMB;
  245. if (private_memory_mb > ZoneNodeManager.NodeConfig.SYSTEM_MAX_HEAP_SIZE_MB)
  246. {
  247. string msg = string.Format("CreateZone failed : Private memory {0}MB out of range {1}MB", private_memory_mb, ZoneNodeManager.NodeConfig.SYSTEM_MAX_HEAP_SIZE_MB);
  248. log.Error(msg);
  249. throw new OutOfMemoryException(msg);
  250. }
  251. XmdsZoneNode node = new XmdsZoneNode(this, instanceId, gameServerId);
  252. node.Node.Callback = IceManager.instance().getCallback(gameServerId);
  253. lock (nodes)
  254. {
  255. if (nodes.ContainsKey(instanceId))
  256. {
  257. throw new Exception(string.Format("node instance id ({0}) already exist!", instanceId));
  258. }
  259. nodes.Add(node.InstanceID, node);
  260. }
  261. {
  262. node.Start(mapTemplateId, data, (z) =>
  263. {
  264. //监控日志
  265. dynamic logEvent = new ExpandoObject();
  266. logEvent.type = "createZone";
  267. logEvent.mapTemplateId = mapTemplateId;
  268. logEvent.instanceId = instanceId;
  269. monitorLog.Debug(logEvent);
  270. });
  271. }
  272. JSGServerProfile.RecordZoneCreate(mapTemplateId, (int)(TimeUtil.GetTimestampMS() - lstart));
  273. }
  274. else
  275. {
  276. log.Warn("玩家pvp状态创建场景:" + playerId + ", " + mapTemplateId + ", " + data);
  277. }
  278. cb(resCode);
  279. }
  280. public void DestroyZone(string instanceId, Action<Exception, object> cb)
  281. {
  282. XmdsZoneNode node;
  283. lock (nodes)
  284. {
  285. node = this.nodes.RemoveByKey(instanceId);
  286. }
  287. if (node != null)
  288. {
  289. //删除场景实例的所有玩家
  290. lock (players)
  291. {
  292. node.Node.ForEachPlayers((client) =>
  293. {
  294. var player = players.Get(client.PlayerUUID);
  295. if (player != null)
  296. {
  297. if (client.Node == player.Node)
  298. {
  299. players.RemoveByKey(client.PlayerUUID);
  300. }
  301. //else
  302. //{
  303. // log.Warn(client.Node + " destory zone player " + client.PlayerUUID + " in " + player.Node);
  304. //}
  305. }
  306. });
  307. }
  308. node.Stop((z) =>
  309. {
  310. //监控日志
  311. dynamic logEvent = new ExpandoObject();
  312. logEvent.type = "destroyZone";
  313. logEvent.mapTemplateId = z.Node.SceneID;
  314. logEvent.instanceId = instanceId;
  315. monitorLog.Debug(logEvent);
  316. });
  317. }
  318. cb(null, "done");
  319. }
  320. public void DestoryAllZones(Action<Exception, object> cb)
  321. {
  322. lock (nodes)
  323. {
  324. foreach (var p in new List<XmdsZoneNode>(nodes.Values))
  325. {
  326. try
  327. {
  328. this.DestroyZone(p.InstanceID, (e, c) => { });
  329. }
  330. catch (Exception e)
  331. {
  332. log.Error(e);
  333. }
  334. }
  335. }
  336. cb(null, "done");
  337. }
  338. public void PlayerEnter(string playerId, string instanceId, string input, Action<Exception, object> cb)
  339. {
  340. var node = getZoneNode(instanceId);
  341. if (node == null)
  342. {
  343. throw new InstanceNotExistException(instanceId);
  344. }
  345. long private_memory_mb = ProcessPrivateMemoryMB;
  346. if (private_memory_mb > ZoneNodeManager.NodeConfig.SYSTEM_MAX_HEAP_SIZE_MB)
  347. {
  348. string msg = string.Format("PlayerEnter failed : Private memory {0}MB out of range {1}MB", private_memory_mb, ZoneNodeManager.NodeConfig.SYSTEM_MAX_HEAP_SIZE_MB);
  349. log.Error(msg);
  350. throw new OutOfMemoryException(msg);
  351. }
  352. dynamic data = Json.Decode(input);
  353. string connectServerId = (string)data.connectServerId;
  354. var session = FastStream.instance().GetSessionByID(connectServerId);
  355. if (session == null)
  356. {
  357. throw new Exception("ConnectServerId not exist : " + connectServerId);
  358. }
  359. string uid = (string)data.uid;
  360. bool robot = data.robot != null ? (bool)data.robot : false;
  361. XmdsPlayerEnter enter = null;
  362. if (data["tempData"] != null)
  363. {
  364. int posX = 0;
  365. int posY = 0;
  366. float direction = 0;
  367. string flagName = null;
  368. flagName = data.tempData.flag;
  369. if (data.tempData["x"] != null)
  370. {
  371. posX = (int)data.tempData.x;
  372. }
  373. if (data.tempData["y"] != null)
  374. {
  375. posY = (int)data.tempData.y;
  376. }
  377. if (data.tempData["direction"] != null)
  378. {
  379. direction = (float)data.tempData.direction;
  380. }
  381. int unitTemplateID = (int)data.unitTemplateID;
  382. XmdsUnitProperties unitprop = new XmdsUnitProperties();
  383. XmdsUnitData unitData = XmdsPlayerUtil.instance().createXmdsUnitData(data, playerId);
  384. unitprop.ServerData = unitData;
  385. enter = new XmdsPlayerEnter();
  386. enter.Pos = new CommonLang.Vector.Vector2(posX, posY);
  387. enter.direction = direction;
  388. enter.FlagName = flagName;
  389. enter.UnitData = new CommonAI.ZoneServer.CreateUnitInfoR2B();
  390. enter.UnitData.Force = unitData.BaseInfo.force;
  391. enter.UnitData.alliesForce = unitData.BaseInfo.alliesForce;
  392. enter.UnitData.StartFlag = null;
  393. enter.UnitData.UnitTemplateID = unitTemplateID;
  394. enter.UnitData.UnitPropData = unitprop;
  395. foreach (dynamic obj in data.tasks)
  396. {
  397. int taskId = obj.QuestID;
  398. int state = obj.State;
  399. XmdsQuestData qData = new XmdsQuestData();
  400. qData.TaskID = taskId.ToString();
  401. qData.TaskState = state.ToString();
  402. foreach (dynamic attr in obj.Attributes)
  403. {
  404. qData.AddAttribute(attr.key, attr.value);
  405. }
  406. unitData.Tasks.Add(qData);
  407. }
  408. foreach (dynamic obj in data.flags)
  409. {
  410. XmdsQuestFlag flag = new XmdsQuestFlag();
  411. flag.FlagName = obj[0];
  412. flag.FlagValue = obj[1];
  413. unitData.QuestFlags.Add(flag);
  414. }
  415. unitData.PlayerEntered = data.playerEntered;
  416. }
  417. XmdsPlayer player = new XmdsPlayer(this, playerId, uid, session, node);
  418. lock (players)
  419. {
  420. if (players.ContainsKey(playerId))
  421. {
  422. this.players[playerId] = player;
  423. //string oldInstanceId = players.Get(playerId).InstanceId;
  424. //PlayerLeave(playerId, oldInstanceId, false, (err, ret) =>
  425. //{
  426. // if (err != null)
  427. // {
  428. // log.Error(err);
  429. // }
  430. //});
  431. }
  432. else
  433. {
  434. this.players.Add(playerId, player);
  435. }
  436. node.Node.OnPlayerEnter(player, enter,
  437. (c) =>
  438. {
  439. player.BindingActor.IsRobot = robot;
  440. //监控日志
  441. dynamic logEvent = new ExpandoObject();
  442. logEvent.type = "playerEnter";
  443. logEvent.mapTemplateId = node.Node.SceneID;
  444. logEvent.instanceId = instanceId;
  445. logEvent.playerId = playerId;
  446. monitorLog.Debug(logEvent);
  447. //cb(null, "done");
  448. },
  449. (e) =>
  450. {
  451. log.Error(e.Message, e);
  452. lock (this.players) { this.players.RemoveByKey(playerId); }
  453. //cb(e, null);
  454. });
  455. }
  456. cb(null, "done");
  457. }
  458. public void PlayerLeave(string playerId, string instanceId, bool keepObject, Action<Exception, object> cb)
  459. {
  460. XmdsPlayer player = null;
  461. try
  462. {
  463. lock (players)
  464. {
  465. player = players.Get(playerId);
  466. if (player != null)
  467. {
  468. if (instanceId.Equals(player.InstanceId))
  469. {
  470. this.players.RemoveByKey(playerId);
  471. }
  472. else
  473. {
  474. log.Warn(instanceId + " leave player " + player.InstanceId + player.Node);
  475. }
  476. }
  477. }
  478. var node = getZoneNode(instanceId);
  479. if (node == null)
  480. {
  481. //throw new InstanceNotExistException(instanceId);
  482. cb(null, "done");
  483. return;
  484. }
  485. //玩家已经离开场景
  486. //if (player == null)
  487. //{
  488. // throw new PlayerNotExistException(playerId);
  489. //}
  490. node.Node.OnPlayerLeave(player != null ? player.BindingActor : node.Node.GetPlayer(playerId),
  491. (c) =>
  492. {
  493. //监控日志
  494. dynamic logEvent = new ExpandoObject();
  495. logEvent.type = "playerLeave";
  496. logEvent.mapTemplateId = node.Node.SceneID;
  497. logEvent.playerId = playerId;
  498. monitorLog.Debug(logEvent);
  499. //cb(null, "done");
  500. },
  501. (e) =>
  502. {
  503. //cb(e, null);
  504. log.Error(e.Message, e);
  505. },
  506. keepObject);
  507. cb(null, "done");
  508. }
  509. catch (Exception e)
  510. {
  511. log.Error(e.Message, e);
  512. cb(e, null);
  513. }
  514. finally
  515. {
  516. if (player != null) player.Dispose();
  517. }
  518. }
  519. public void PlayerNetStateChanged(string playerId, string state)
  520. {
  521. var player = getPlayer(playerId);
  522. if (player == null)
  523. {
  524. log.Error("player not exist :" + playerId + " PlayerNetStateChanged");
  525. return;
  526. }
  527. var node = player.Node;
  528. if (node == null)
  529. {
  530. log.Error("zone node not exist :" + player.InstanceId + " PlayerNetStateChanged");
  531. return;
  532. }
  533. log.Debug("player:" + playerId + " PlayerNetStateChanged " + state);
  534. node.OnPlayerNetStateChanged(player, state);
  535. }
  536. public void PlayerReceive(string playerId, byte[] msg)
  537. {
  538. var player = getPlayer(playerId);
  539. if (player == null)
  540. {
  541. log.Debug("fast stream receive not exist player : " + playerId);
  542. return;
  543. }
  544. var node = player.Node;
  545. if (node == null)
  546. {
  547. log.Error("zone node not exist : " + player.InstanceId + " fast stream receive");
  548. return;
  549. }
  550. //log.Debug("player:" + playerId + " socket.receive");
  551. node.OnPlayerReceivedMessage(player, msg);
  552. }
  553. public void GetAllPlayerCount(Action<Exception, int> cb)
  554. {
  555. int count = PlayerCount;
  556. cb(null, count);
  557. }
  558. public void ClearAllPlayers(Action<Exception, object> cb)
  559. {
  560. try
  561. {
  562. lock (players)
  563. {
  564. foreach (var p in new List<XmdsPlayer>(players.Values))
  565. {
  566. this.PlayerLeave(p.PlayerUUID, p.InstanceId, false, (e, c) => { });
  567. }
  568. }
  569. cb(null, "done");
  570. }
  571. catch (Exception e)
  572. {
  573. cb(e, null);
  574. }
  575. }
  576. public void GetAllPlayers(Action<Exception, object> cb)
  577. {
  578. try
  579. {
  580. var ret = new List<string>();
  581. lock (players)
  582. {
  583. foreach (var p in new List<XmdsPlayer>(players.Values))
  584. {
  585. ret.Add(p.PlayerUUID);
  586. }
  587. }
  588. cb(null, ret.ToArray());
  589. }
  590. catch (Exception e)
  591. {
  592. cb(e, null);
  593. }
  594. }
  595. public void GetServerState(string serverID, Action<Exception, object> cb)
  596. {
  597. try
  598. {
  599. int zc = ZoneNodeCount;
  600. int pc = PlayerCount;
  601. int ActiveObjectCount = InstanceZoneObject.ActiveObjectCount;
  602. int AllocObjectCount = InstanceZoneObject.AllocObjectCount;
  603. int ActiveZoneCount = InstanceZone.ActiveZoneCount;
  604. int AllocZoneCount = InstanceZone.AllocZoneCount;
  605. float w = zc * 1000 + pc;
  606. int flag = IceManager.instance().getCallback(serverID) == null ? 0 : 1;
  607. dynamic ret = new
  608. {
  609. weight = w,
  610. memory = ProcessPrivateMemoryMB,
  611. zone_count = zc,
  612. player_count = pc,
  613. ActiveObjectCount = ActiveObjectCount,
  614. AllocObjectCount = AllocObjectCount,
  615. ActiveZoneCount = ActiveZoneCount,
  616. AllocZoneCount = AllocZoneCount,
  617. flag = flag,
  618. };
  619. cb(null, ret);
  620. }
  621. catch (Exception e)
  622. {
  623. log.Error("GetServerState failed, error: " + e.Message, e);
  624. cb(e, e.Message);
  625. }
  626. }
  627. public void RegisterGameServer(int serverId, int crossId, Action<Exception, string> cb)
  628. {
  629. try
  630. {
  631. var str = ConfigurationManager.AppSettings["game.server.id"];
  632. if (!string.IsNullOrEmpty(str) && int.Parse(str) != serverId && int.Parse(str) != crossId)
  633. {
  634. cb(null, "1");
  635. return;
  636. }
  637. }
  638. catch (Exception e)
  639. {
  640. cb(e, "2");
  641. return;
  642. }
  643. cb(null, S_BSVersion);
  644. }
  645. //获取单位血量
  646. public void GetUnitHP(string instanceId, int objectId, Action<Exception, int> cb)
  647. {
  648. try
  649. {
  650. var node = getZoneNode(instanceId);
  651. if (node == null)
  652. {
  653. cb(null, -1);
  654. return;
  655. }
  656. InstanceUnit unit = node.Node.Zone.getUnit((uint)objectId);
  657. if (unit == null)
  658. {
  659. cb(null, -2);
  660. return;
  661. }
  662. cb(null, unit.CurrentHP);
  663. }
  664. catch (Exception e)
  665. {
  666. cb(e, -10);
  667. return;
  668. }
  669. }
  670. //----------------------------------------------------------------------------------------------
  671. }
  672. }