您最多选择25个主题 主题必须以字母或数字开头,可以包含连字符 (-),并且长度不得超过35个字符

ProcessMode.java 14KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341
  1. /*
  2. * Copyright (c) 2006-2008 Chris Smith, Shane Mc Cormack, Gregory Holmes
  3. *
  4. * Permission is hereby granted, free of charge, to any person obtaining a copy
  5. * of this software and associated documentation files (the "Software"), to deal
  6. * in the Software without restriction, including without limitation the rights
  7. * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
  8. * copies of the Software, and to permit persons to whom the Software is
  9. * furnished to do so, subject to the following conditions:
  10. *
  11. * The above copyright notice and this permission notice shall be included in
  12. * all copies or substantial portions of the Software.
  13. *
  14. * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  15. * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  16. * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
  17. * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
  18. * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
  19. * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
  20. * SOFTWARE.
  21. */
  22. package com.dmdirc.parser;
  23. import com.dmdirc.parser.callbacks.CallbackOnChannelSingleModeChanged;
  24. import com.dmdirc.parser.callbacks.CallbackOnChannelNonUserModeChanged;
  25. import com.dmdirc.parser.callbacks.CallbackOnChannelModeChanged;
  26. import com.dmdirc.parser.callbacks.CallbackOnChannelUserModeChanged;
  27. import com.dmdirc.parser.callbacks.CallbackOnUserModeChanged;
  28. import com.dmdirc.parser.callbacks.CallbackOnUserModeDiscovered;
  29. import java.util.Calendar;
  30. /**
  31. * Process a Mode line.
  32. */
  33. public class ProcessMode extends IRCProcessor {
  34. /**
  35. * Process a Mode Line.
  36. *
  37. * @param sParam Type of line to process ("MODE", "324")
  38. * @param token IRCTokenised line to process
  39. */
  40. @Override
  41. public void process(String sParam, String[] token) {
  42. String[] sModestr;
  43. String sChannelName;
  44. if (sParam.equals("324")) {
  45. sChannelName = token[3];
  46. sModestr = new String[token.length-4];
  47. System.arraycopy(token, 4, sModestr, 0, token.length-4);
  48. } else if (sParam.equals("221")) {
  49. processUserMode(sParam, token, new String[]{token[token.length-1]}, true);
  50. return;
  51. } else {
  52. sChannelName = token[2];
  53. sModestr = new String[token.length-3];
  54. System.arraycopy(token, 3, sModestr, 0, token.length-3);
  55. }
  56. if (!isValidChannelName(sChannelName)) { processUserMode(sParam, token, sModestr, false); }
  57. else { processChanMode(sParam, token, sModestr, sChannelName); }
  58. }
  59. /**
  60. * Method to trim spaces from strings
  61. *
  62. * @param str String to trim
  63. * @return String without spaces on the ends
  64. */
  65. private String trim(String str) { return str.trim(); }
  66. /**
  67. * Process Chan modes.
  68. *
  69. * @param sParam String representation of parameter to parse
  70. * @param token IRCTokenised Array of the incomming line
  71. * @param sModestr The modes and params
  72. * @param sChannelName Channel these modes are for
  73. */
  74. public void processChanMode(String sParam, String token[], String sModestr[], String sChannelName) {
  75. StringBuilder sFullModeStr = new StringBuilder();
  76. String sNonUserModeStr = "";
  77. String sNonUserModeStrParams = "";
  78. String sModeParam;
  79. String sTemp;
  80. int nParam = 1;
  81. long nTemp = 0, nValue = 0, nCurrent = 0;
  82. boolean bPositive = true, bBooleanMode = true;
  83. char cPositive = '+';
  84. ChannelInfo iChannel;
  85. ChannelClientInfo iChannelClientInfo;
  86. ClientInfo iClient;
  87. ChannelClientInfo setterCCI;
  88. CallbackOnChannelSingleModeChanged cbSingle = null;
  89. CallbackOnChannelNonUserModeChanged cbNonUser = null;
  90. if (!sParam.equals("324")) {
  91. cbSingle = (CallbackOnChannelSingleModeChanged) getCallbackManager().getCallbackType("OnChannelSingleModeChanged");
  92. cbNonUser = (CallbackOnChannelNonUserModeChanged) getCallbackManager().getCallbackType("OnChannelNonUserModeChanged");
  93. }
  94. iChannel = getChannelInfo(sChannelName);
  95. if (iChannel == null) {
  96. // callErrorInfo(new ParserError(ParserError.ERROR_WARNING, "Got modes for channel ("+sChannelName+") that I am not on.", myParser.getLastLine()));
  97. // iChannel = new ChannelInfo(myParser, sChannelName);
  98. // myParser.addChannel(iChannel);
  99. return;
  100. }
  101. // Get the current channel modes
  102. if (!sParam.equals("324")) { nCurrent = iChannel.getMode(); }
  103. setterCCI = iChannel.getUser(token[0]);
  104. if (IRCParser.ALWAYS_UPDATECLIENT && setterCCI != null) {
  105. // Facilitate dmdirc formatter
  106. if (setterCCI.getClient().getHost().isEmpty()) {setterCCI.getClient().setUserBits(token[0],false); }
  107. }
  108. // Loop through the mode string, and add/remove modes/params where they are needed
  109. for (int i = 0; i < sModestr[0].length(); ++i) {
  110. Character cMode = sModestr[0].charAt(i);
  111. if (cMode.equals(":".charAt(0))) { continue; }
  112. sNonUserModeStr = sNonUserModeStr+cMode;
  113. if (cMode.equals("+".charAt(0))) { cPositive = '+'; bPositive = true; }
  114. else if (cMode.equals("-".charAt(0))) { cPositive = '-'; bPositive = false; }
  115. else {
  116. if (myParser.hChanModesBool.containsKey(cMode)) { nValue = myParser.hChanModesBool.get(cMode); bBooleanMode = true; }
  117. else if (myParser.hChanModesOther.containsKey(cMode)) { nValue = myParser.hChanModesOther.get(cMode); bBooleanMode = false; }
  118. else if (myParser.hPrefixModes.containsKey(cMode)) {
  119. // (de) OP/Voice someone
  120. sModeParam = sModestr[nParam++];
  121. nValue = myParser.hPrefixModes.get(cMode);
  122. callDebugInfo(IRCParser.DEBUG_INFO, "User Mode: %c / %d [%s] {Positive: %b}",cMode, nValue, sModeParam, bPositive);
  123. iChannelClientInfo = iChannel.getUser(sModeParam);
  124. if (iChannelClientInfo == null) {
  125. // Client not known?
  126. // callErrorInfo(new ParserError(ParserError.ERROR_WARNING, "Got mode for client not known on channel - Added", myParser.getLastLine()));
  127. iClient = getClientInfo(sModeParam);
  128. if (iClient == null) {
  129. // callErrorInfo(new ParserError(ParserError.ERROR_WARNING, "Got mode for client not known at all - Added", myParser.getLastLine()));
  130. iClient = new ClientInfo(myParser, sModeParam);
  131. myParser.addClient(iClient);
  132. }
  133. iChannelClientInfo = iChannel.addClient(iClient);
  134. }
  135. callDebugInfo(IRCParser.DEBUG_INFO, "\tOld Mode Value: %d",iChannelClientInfo.getChanMode());
  136. if (bPositive) { iChannelClientInfo.setChanMode(iChannelClientInfo.getChanMode() | nValue); sTemp = "+"; }
  137. else { iChannelClientInfo.setChanMode(iChannelClientInfo.getChanMode() ^ (iChannelClientInfo.getChanMode() & nValue)); sTemp = "-"; }
  138. sTemp = sTemp+cMode;
  139. callChannelUserModeChanged(iChannel, iChannelClientInfo, setterCCI, token[0], sTemp);
  140. continue;
  141. } else {
  142. // unknown mode - add as boolean
  143. // callErrorInfo(new ParserError(ParserError.ERROR_WARNING, "Got unknown mode "+cMode+" - Added as boolean mode", myParser.getLastLine()));
  144. myParser.hChanModesBool.put(cMode,myParser.nNextKeyCMBool);
  145. nValue = myParser.nNextKeyCMBool;
  146. bBooleanMode = true;
  147. myParser.nNextKeyCMBool = myParser.nNextKeyCMBool*2;
  148. }
  149. if (bBooleanMode) {
  150. callDebugInfo(IRCParser.DEBUG_INFO, "Boolean Mode: %c [%d] {Positive: %b}",cMode, nValue, bPositive);
  151. if (bPositive) { nCurrent = nCurrent | nValue; }
  152. else { nCurrent = nCurrent ^ (nCurrent & nValue); }
  153. } else {
  154. if ((bPositive || nValue == IRCParser.MODE_LIST || ((nValue & IRCParser.MODE_UNSET) == IRCParser.MODE_UNSET)) && (sModestr.length <= nParam)) {
  155. myParser.callErrorInfo(new ParserError(ParserError.ERROR_FATAL + ParserError.ERROR_USER, "Broken Modes. Parameter required but not given.", myParser.getLastLine()));
  156. }
  157. if (nValue == IRCParser.MODE_LIST) {
  158. // List Mode
  159. sModeParam = sModestr[nParam++];
  160. sNonUserModeStrParams = sNonUserModeStrParams+" "+sModeParam;
  161. nTemp = (Calendar.getInstance().getTimeInMillis() / 1000);
  162. iChannel.setListModeParam(cMode, new ChannelListModeItem(sModeParam, token[0], nTemp ), bPositive);
  163. callDebugInfo(IRCParser.DEBUG_INFO, "List Mode: %c [%s] {Positive: %b}",cMode, sModeParam, bPositive);
  164. if (cbSingle != null) { cbSingle.call(iChannel, setterCCI, token[0], cPositive+cMode+" "+sModeParam ); }
  165. } else {
  166. // Mode with a parameter
  167. if (bPositive) {
  168. // +Mode - always needs a parameter to set
  169. sModeParam = sModestr[nParam++];
  170. sNonUserModeStrParams = sNonUserModeStrParams+" "+sModeParam;
  171. callDebugInfo(IRCParser.DEBUG_INFO, "Set Mode: %c [%s] {Positive: %b}",cMode, sModeParam, bPositive);
  172. iChannel.setModeParam(cMode,sModeParam);
  173. if (cbSingle != null) { cbSingle.call(iChannel, setterCCI, token[0], cPositive+cMode+" "+sModeParam ); }
  174. } else {
  175. // -Mode - parameter isn't always needed, we need to check
  176. if ((nValue & IRCParser.MODE_UNSET) == IRCParser.MODE_UNSET) {
  177. sModeParam = sModestr[nParam++];
  178. sNonUserModeStrParams = sNonUserModeStrParams+" "+sModeParam;
  179. } else {
  180. sModeParam = "";
  181. }
  182. callDebugInfo(IRCParser.DEBUG_INFO, "Unset Mode: %c [%s] {Positive: %b}",cMode, sModeParam, bPositive);
  183. iChannel.setModeParam(cMode,"");
  184. if (cbSingle != null) { cbSingle.call(iChannel, setterCCI, token[0], trim(cPositive+cMode+" "+sModeParam) ); }
  185. }
  186. }
  187. }
  188. }
  189. }
  190. // Call Callbacks
  191. for (int i = 0; i < sModestr.length; ++i) { sFullModeStr.append(sModestr[i]).append(" "); }
  192. iChannel.setMode(nCurrent);
  193. if (sParam.equals("324")) { callChannelModeChanged(iChannel, null, "", sFullModeStr.toString().trim()); }
  194. else { callChannelModeChanged(iChannel, setterCCI, token[0], sFullModeStr.toString().trim()); }
  195. if (cbNonUser != null) { cbNonUser.call(iChannel, setterCCI, token[0], trim(sNonUserModeStr+sNonUserModeStrParams)); }
  196. }
  197. /**
  198. * Process user modes.
  199. *
  200. * @param sParam String representation of parameter to parse
  201. * @param token IRCTokenised Array of the incomming line
  202. * @param clearOldModes Clear old modes before applying these modes (used by 221)
  203. */
  204. private void processUserMode(String sParam, String token[], String sModestr[], boolean clearOldModes) {
  205. long nCurrent = 0, nValue = 0;
  206. boolean bPositive = true;
  207. ClientInfo iClient;
  208. iClient = getClientInfo(token[2]);
  209. if (iClient == null) { return; }
  210. if (clearOldModes) {
  211. nCurrent = 0;
  212. } else {
  213. nCurrent = iClient.getUserMode();
  214. }
  215. for (int i = 0; i < sModestr[0].length(); ++i) {
  216. Character cMode = sModestr[0].charAt(i);
  217. if (cMode.equals("+".charAt(0))) { bPositive = true; }
  218. else if (cMode.equals("-".charAt(0))) { bPositive = false; }
  219. else if (cMode.equals(":".charAt(0))) { continue; }
  220. else {
  221. if (myParser.hUserModes.containsKey(cMode)) { nValue = myParser.hUserModes.get(cMode); }
  222. else {
  223. // Unknown mode
  224. callErrorInfo(new ParserError(ParserError.ERROR_WARNING, "Got unknown user mode "+cMode+" - Added", myParser.getLastLine()));
  225. myParser.hUserModes.put(cMode,myParser.nNextKeyUser);
  226. nValue = myParser.nNextKeyUser;
  227. myParser.nNextKeyUser = myParser.nNextKeyUser*2;
  228. }
  229. // Usermodes are always boolean
  230. callDebugInfo(IRCParser.DEBUG_INFO, "User Mode: %c [%d] {Positive: %b}",cMode, nValue, bPositive);
  231. if (bPositive) { nCurrent = nCurrent | nValue; }
  232. else { nCurrent = nCurrent ^ (nCurrent & nValue); }
  233. }
  234. }
  235. iClient.setUserMode(nCurrent);
  236. if (sParam.equals("221")) {
  237. callUserModeDiscovered(iClient, sModestr[0]);
  238. } else {
  239. callUserModeChanged(iClient, token[0], sModestr[0]);
  240. }
  241. }
  242. /**
  243. * Callback to all objects implementing the ChannelModeChanged Callback.
  244. *
  245. * @see IChannelModeChanged
  246. * @param cChannel Channel where modes were changed
  247. * @param cChannelClient Client chaning the modes (null if server)
  248. * @param sHost Host doing the mode changing (User host or server name)
  249. * @param sModes Exact String parsed
  250. * @return true if a method was called, false otherwise
  251. */
  252. protected boolean callChannelModeChanged(ChannelInfo cChannel, ChannelClientInfo cChannelClient, String sHost, String sModes) {
  253. return ((CallbackOnChannelModeChanged) getCallbackManager().getCallbackType("OnChannelModeChanged")).call(cChannel, cChannelClient, sHost, sModes);
  254. }
  255. /**
  256. * Callback to all objects implementing the ChannelUserModeChanged Callback.
  257. *
  258. * @see IChannelUserModeChanged
  259. * @param cChannel Channel where modes were changed
  260. * @param cChangedClient Client being changed
  261. * @param cSetByClient Client chaning the modes (null if server)
  262. * @param sMode String representing mode change (ie +o)
  263. * @param sHost Host doing the mode changing (User host or server name)
  264. * @return true if a method was called, false otherwise
  265. */
  266. protected boolean callChannelUserModeChanged(ChannelInfo cChannel, ChannelClientInfo cChangedClient, ChannelClientInfo cSetByClient, String sHost, String sMode) {
  267. return ((CallbackOnChannelUserModeChanged) getCallbackManager().getCallbackType("OnChannelUserModeChanged")).call(cChannel, cChangedClient, cSetByClient, sHost, sMode);
  268. }
  269. /**
  270. * Callback to all objects implementing the UserModeChanged Callback.
  271. *
  272. * @see IUserModeChanged
  273. * @param cClient Client that had the mode changed (almost always us)
  274. * @param sSetby Host that set the mode (us or servername)
  275. * @param sModes The modes set.
  276. * @return true if a method was called, false otherwise
  277. */
  278. protected boolean callUserModeChanged(ClientInfo cClient, String sSetby, String sModes) {
  279. return ((CallbackOnUserModeChanged) getCallbackManager().getCallbackType("OnUserModeChanged")).call(cClient, sSetby, sModes);
  280. }
  281. /**
  282. * Callback to all objects implementing the UserModeDiscovered Callback.
  283. *
  284. * @see IUserModeDiscovered
  285. * @param cClient Client that had the mode changed (almost always us)
  286. * @param sModes The modes set.
  287. * @return true if a method was called, false otherwise
  288. */
  289. protected boolean callUserModeDiscovered(ClientInfo cClient, String sModes) {
  290. return ((CallbackOnUserModeDiscovered)getCallbackManager().getCallbackType("OnUserModeDiscovered")).call(cClient, sModes);
  291. }
  292. /**
  293. * What does this IRCProcessor handle.
  294. *
  295. * @return String[] with the names of the tokens we handle.
  296. */
  297. @Override
  298. public String[] handles() {
  299. return new String[]{"MODE", "324", "221"};
  300. }
  301. /**
  302. * Create a new instance of the IRCProcessor Object.
  303. *
  304. * @param parser IRCParser That owns this IRCProcessor
  305. * @param manager ProcessingManager that is in charge of this IRCProcessor
  306. */
  307. protected ProcessMode (IRCParser parser, ProcessingManager manager) { super(parser, manager); }
  308. }