You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

ProcessMode.java 17KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346
  1. /*
  2. * Copyright (c) 2006-2009 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.irc;
  23. import com.dmdirc.parser.common.ChannelListModeItem;
  24. import com.dmdirc.parser.common.ParserError;
  25. import com.dmdirc.parser.interfaces.ChannelClientInfo;
  26. import com.dmdirc.parser.interfaces.ChannelInfo;
  27. import com.dmdirc.parser.interfaces.ClientInfo;
  28. import com.dmdirc.parser.interfaces.callbacks.ChannelModeChangeListener;
  29. import com.dmdirc.parser.interfaces.callbacks.ChannelNonUserModeChangeListener;
  30. import com.dmdirc.parser.interfaces.callbacks.ChannelSingleModeChangeListener;
  31. import com.dmdirc.parser.interfaces.callbacks.ChannelUserModeChangeListener;
  32. import com.dmdirc.parser.interfaces.callbacks.UserModeChangeListener;
  33. import com.dmdirc.parser.interfaces.callbacks.UserModeDiscoveryListener;
  34. import com.dmdirc.parser.common.CallbackObject;
  35. import java.util.Calendar;
  36. /**
  37. * Process a Mode line.
  38. */
  39. public class ProcessMode extends IRCProcessor {
  40. /**
  41. * Process a Mode Line.
  42. *
  43. * @param sParam Type of line to process ("MODE", "324")
  44. * @param token IRCTokenised line to process
  45. */
  46. @Override
  47. public void process(String sParam, String[] token) {
  48. String[] sModestr;
  49. String sChannelName;
  50. if (sParam.equals("324")) {
  51. sChannelName = token[3];
  52. sModestr = new String[token.length-4];
  53. System.arraycopy(token, 4, sModestr, 0, token.length-4);
  54. } else if (sParam.equals("221")) {
  55. processUserMode(sParam, token, new String[]{token[token.length-1]}, true);
  56. return;
  57. } else {
  58. sChannelName = token[2];
  59. sModestr = new String[token.length-3];
  60. System.arraycopy(token, 3, sModestr, 0, token.length-3);
  61. }
  62. if (!isValidChannelName(sChannelName)) { processUserMode(sParam, token, sModestr, false); }
  63. else { processChanMode(sParam, token, sModestr, sChannelName); }
  64. }
  65. /**
  66. * Method to trim spaces from strings
  67. *
  68. * @param str String to trim
  69. * @return String without spaces on the ends
  70. */
  71. private String trim(String str) { return str.trim(); }
  72. /**
  73. * Process Chan modes.
  74. *
  75. * @param sParam String representation of parameter to parse
  76. * @param token IRCTokenised Array of the incomming line
  77. * @param sModestr The modes and params
  78. * @param sChannelName Channel these modes are for
  79. */
  80. public void processChanMode(String sParam, String token[], String sModestr[], String sChannelName) {
  81. StringBuilder sFullModeStr = new StringBuilder();
  82. String sNonUserModeStr = "";
  83. String sNonUserModeStrParams = "";
  84. String sModeParam;
  85. String sTemp;
  86. int nParam = 1;
  87. long nTemp = 0, nValue = 0, nCurrent = 0;
  88. boolean bPositive = true, bBooleanMode = true;
  89. char cPositive = '+';
  90. IRCChannelInfo iChannel;
  91. IRCChannelClientInfo iChannelClientInfo;
  92. IRCClientInfo iClient;
  93. IRCChannelClientInfo setterCCI;
  94. CallbackObject cbSingle = null;
  95. CallbackObject cbNonUser = null;
  96. if (!sParam.equals("324")) {
  97. cbSingle = getCallbackManager().getCallbackType(ChannelSingleModeChangeListener.class);
  98. cbNonUser = getCallbackManager().getCallbackType(ChannelNonUserModeChangeListener.class);
  99. }
  100. iChannel = getChannel(sChannelName);
  101. if (iChannel == null) {
  102. // callErrorInfo(new ParserError(ParserError.ERROR_WARNING, "Got modes for channel ("+sChannelName+") that I am not on.", myParser.getLastLine()));
  103. // iChannel = new ChannelInfo(myParser, sChannelName);
  104. // myParser.addChannel(iChannel);
  105. return;
  106. }
  107. // Get the current channel modes
  108. if (!sParam.equals("324")) { nCurrent = iChannel.getMode(); }
  109. setterCCI = iChannel.getChannelClient(token[0]);
  110. if (IRCParser.ALWAYS_UPDATECLIENT && setterCCI != null) {
  111. // Facilitate dmdirc formatter
  112. if (setterCCI.getClient().getHostname().isEmpty()) {setterCCI.getClient().setUserBits(token[0],false); }
  113. }
  114. // Loop through the mode string, and add/remove modes/params where they are needed
  115. for (int i = 0; i < sModestr[0].length(); ++i) {
  116. Character cMode = sModestr[0].charAt(i);
  117. if (cMode.equals(":".charAt(0))) { continue; }
  118. sNonUserModeStr = sNonUserModeStr+cMode;
  119. if (cMode.equals("+".charAt(0))) { cPositive = '+'; bPositive = true; }
  120. else if (cMode.equals("-".charAt(0))) { cPositive = '-'; bPositive = false; }
  121. else {
  122. if (myParser.chanModesBool.containsKey(cMode)) { nValue = myParser.chanModesBool.get(cMode); bBooleanMode = true; }
  123. else if (myParser.chanModesOther.containsKey(cMode)) { nValue = myParser.chanModesOther.get(cMode); bBooleanMode = false; }
  124. else if (myParser.prefixModes.containsKey(cMode)) {
  125. // (de) OP/Voice someone
  126. if (sModestr.length <= nParam) {
  127. myParser.callErrorInfo(new ParserError(ParserError.ERROR_FATAL + ParserError.ERROR_USER, "Broken Modes. Parameter required but not given.", myParser.getLastLine()));
  128. return;
  129. }
  130. sModeParam = sModestr[nParam++];
  131. nValue = myParser.prefixModes.get(cMode);
  132. callDebugInfo(IRCParser.DEBUG_INFO, "User Mode: %c / %d [%s] {Positive: %b}",cMode, nValue, sModeParam, bPositive);
  133. iChannelClientInfo = iChannel.getChannelClient(sModeParam);
  134. if (iChannelClientInfo == null) {
  135. // Client not known?
  136. iClient = getClientInfo(sModeParam);
  137. if (iClient == null) {
  138. iClient = new IRCClientInfo(myParser, sModeParam);
  139. myParser.addClient(iClient);
  140. }
  141. iChannelClientInfo = iChannel.addClient(iClient);
  142. }
  143. callDebugInfo(IRCParser.DEBUG_INFO, "\tOld Mode Value: %d",iChannelClientInfo.getChanMode());
  144. if (bPositive) { iChannelClientInfo.setChanMode(iChannelClientInfo.getChanMode() | nValue); sTemp = "+"; }
  145. else { iChannelClientInfo.setChanMode(iChannelClientInfo.getChanMode() ^ (iChannelClientInfo.getChanMode() & nValue)); sTemp = "-"; }
  146. sTemp = sTemp+cMode;
  147. callChannelUserModeChanged(iChannel, iChannelClientInfo, setterCCI, token[0], sTemp);
  148. continue;
  149. } else {
  150. // unknown mode - add as boolean
  151. myParser.chanModesBool.put(cMode,myParser.nextKeyCMBool);
  152. nValue = myParser.nextKeyCMBool;
  153. bBooleanMode = true;
  154. myParser.nextKeyCMBool = myParser.nextKeyCMBool*2;
  155. }
  156. if (bBooleanMode) {
  157. callDebugInfo(IRCParser.DEBUG_INFO, "Boolean Mode: %c [%d] {Positive: %b}",cMode, nValue, bPositive);
  158. if (bPositive) { nCurrent = nCurrent | nValue; }
  159. else { nCurrent = nCurrent ^ (nCurrent & nValue); }
  160. } else {
  161. if ((bPositive || nValue == IRCParser.MODE_LIST || ((nValue & IRCParser.MODE_UNSET) == IRCParser.MODE_UNSET)) && (sModestr.length <= nParam)) {
  162. myParser.callErrorInfo(new ParserError(ParserError.ERROR_FATAL + ParserError.ERROR_USER, "Broken Modes. Parameter required but not given.", myParser.getLastLine()));
  163. continue;
  164. }
  165. if (nValue == IRCParser.MODE_LIST) {
  166. // List Mode
  167. sModeParam = sModestr[nParam++];
  168. sNonUserModeStrParams = sNonUserModeStrParams+" "+sModeParam;
  169. nTemp = (Calendar.getInstance().getTimeInMillis() / 1000);
  170. iChannel.setListModeParam(cMode, new ChannelListModeItem(sModeParam, token[0], nTemp ), bPositive);
  171. callDebugInfo(IRCParser.DEBUG_INFO, "List Mode: %c [%s] {Positive: %b}",cMode, sModeParam, bPositive);
  172. if (cbSingle != null) { cbSingle.call(iChannel, setterCCI, token[0], cPositive+cMode+" "+sModeParam ); }
  173. } else {
  174. // Mode with a parameter
  175. if (bPositive) {
  176. // +Mode - always needs a parameter to set
  177. sModeParam = sModestr[nParam++];
  178. sNonUserModeStrParams = sNonUserModeStrParams+" "+sModeParam;
  179. callDebugInfo(IRCParser.DEBUG_INFO, "Set Mode: %c [%s] {Positive: %b}",cMode, sModeParam, bPositive);
  180. iChannel.setModeParam(cMode,sModeParam);
  181. if (cbSingle != null) { cbSingle.call(iChannel, setterCCI, token[0], cPositive+cMode+" "+sModeParam ); }
  182. } else {
  183. // -Mode - parameter isn't always needed, we need to check
  184. if ((nValue & IRCParser.MODE_UNSET) == IRCParser.MODE_UNSET) {
  185. sModeParam = sModestr[nParam++];
  186. sNonUserModeStrParams = sNonUserModeStrParams+" "+sModeParam;
  187. } else {
  188. sModeParam = "";
  189. }
  190. callDebugInfo(IRCParser.DEBUG_INFO, "Unset Mode: %c [%s] {Positive: %b}",cMode, sModeParam, bPositive);
  191. iChannel.setModeParam(cMode,"");
  192. if (cbSingle != null) { cbSingle.call(iChannel, setterCCI, token[0], trim(cPositive+cMode+" "+sModeParam) ); }
  193. }
  194. }
  195. }
  196. }
  197. }
  198. // Call Callbacks
  199. for (int i = 0; i < sModestr.length; ++i) { sFullModeStr.append(sModestr[i]).append(" "); }
  200. iChannel.setMode(nCurrent);
  201. if (sParam.equals("324")) { callChannelModeChanged(iChannel, null, "", sFullModeStr.toString().trim()); }
  202. else { callChannelModeChanged(iChannel, setterCCI, token[0], sFullModeStr.toString().trim()); }
  203. if (cbNonUser != null) { cbNonUser.call(iChannel, setterCCI, token[0], trim(sNonUserModeStr+sNonUserModeStrParams)); }
  204. }
  205. /**
  206. * Process user modes.
  207. *
  208. * @param sParam String representation of parameter to parse
  209. * @param token IRCTokenised Array of the incomming line
  210. * @param clearOldModes Clear old modes before applying these modes (used by 221)
  211. */
  212. private void processUserMode(String sParam, String token[], String sModestr[], boolean clearOldModes) {
  213. long nCurrent = 0, nValue = 0;
  214. boolean bPositive = true;
  215. IRCClientInfo iClient = getClientInfo(token[2]);
  216. if (iClient == null) { return; }
  217. if (clearOldModes) {
  218. nCurrent = 0;
  219. } else {
  220. nCurrent = iClient.getUserMode();
  221. }
  222. for (int i = 0; i < sModestr[0].length(); ++i) {
  223. Character cMode = sModestr[0].charAt(i);
  224. if (cMode.equals("+".charAt(0))) { bPositive = true; }
  225. else if (cMode.equals("-".charAt(0))) { bPositive = false; }
  226. else if (cMode.equals(":".charAt(0))) { continue; }
  227. else {
  228. if (myParser.userModes.containsKey(cMode)) { nValue = myParser.userModes.get(cMode); }
  229. else {
  230. // Unknown mode
  231. callErrorInfo(new ParserError(ParserError.ERROR_WARNING, "Got unknown user mode "+cMode+" - Added", myParser.getLastLine()));
  232. myParser.userModes.put(cMode,myParser.nNextKeyUser);
  233. nValue = myParser.nNextKeyUser;
  234. myParser.nNextKeyUser = myParser.nNextKeyUser*2;
  235. }
  236. // Usermodes are always boolean
  237. callDebugInfo(IRCParser.DEBUG_INFO, "User Mode: %c [%d] {Positive: %b}",cMode, nValue, bPositive);
  238. if (bPositive) { nCurrent = nCurrent | nValue; }
  239. else { nCurrent = nCurrent ^ (nCurrent & nValue); }
  240. }
  241. }
  242. iClient.setUserMode(nCurrent);
  243. if (sParam.equals("221")) {
  244. callUserModeDiscovered(iClient, sModestr[0]);
  245. } else {
  246. callUserModeChanged(iClient, token[0], sModestr[0]);
  247. }
  248. }
  249. /**
  250. * Callback to all objects implementing the ChannelModeChanged Callback.
  251. *
  252. * @see IChannelModeChanged
  253. * @param cChannel Channel where modes were changed
  254. * @param cChannelClient Client chaning the modes (null if server)
  255. * @param sHost Host doing the mode changing (User host or server name)
  256. * @param sModes Exact String parsed
  257. * @return true if a method was called, false otherwise
  258. */
  259. protected boolean callChannelModeChanged(ChannelInfo cChannel, ChannelClientInfo cChannelClient, String sHost, String sModes) {
  260. return getCallbackManager().getCallbackType(ChannelModeChangeListener.class).call(cChannel, cChannelClient, sHost, sModes);
  261. }
  262. /**
  263. * Callback to all objects implementing the ChannelUserModeChanged Callback.
  264. *
  265. * @see IChannelUserModeChanged
  266. * @param cChannel Channel where modes were changed
  267. * @param cChangedClient Client being changed
  268. * @param cSetByClient Client chaning the modes (null if server)
  269. * @param sMode String representing mode change (ie +o)
  270. * @param sHost Host doing the mode changing (User host or server name)
  271. * @return true if a method was called, false otherwise
  272. */
  273. protected boolean callChannelUserModeChanged(ChannelInfo cChannel, ChannelClientInfo cChangedClient, ChannelClientInfo cSetByClient, String sHost, String sMode) {
  274. return getCallbackManager().getCallbackType(ChannelUserModeChangeListener.class).call(cChannel, cChangedClient, cSetByClient, sHost, sMode);
  275. }
  276. /**
  277. * Callback to all objects implementing the UserModeChanged Callback.
  278. *
  279. * @see IUserModeChanged
  280. * @param cClient Client that had the mode changed (almost always us)
  281. * @param sSetby Host that set the mode (us or servername)
  282. * @param sModes The modes set.
  283. * @return true if a method was called, false otherwise
  284. */
  285. protected boolean callUserModeChanged(ClientInfo cClient, String sSetby, String sModes) {
  286. return getCallbackManager().getCallbackType(UserModeChangeListener.class).call(cClient, sSetby, sModes);
  287. }
  288. /**
  289. * Callback to all objects implementing the UserModeDiscovered Callback.
  290. *
  291. * @see IUserModeDiscovered
  292. * @param cClient Client that had the mode changed (almost always us)
  293. * @param sModes The modes set.
  294. * @return true if a method was called, false otherwise
  295. */
  296. protected boolean callUserModeDiscovered(ClientInfo cClient, String sModes) {
  297. return getCallbackManager().getCallbackType(UserModeDiscoveryListener.class).call(cClient, sModes);
  298. }
  299. /**
  300. * What does this IRCProcessor handle.
  301. *
  302. * @return String[] with the names of the tokens we handle.
  303. */
  304. @Override
  305. public String[] handles() {
  306. return new String[]{"MODE", "324", "221"};
  307. }
  308. /**
  309. * Create a new instance of the IRCProcessor Object.
  310. *
  311. * @param parser IRCParser That owns this IRCProcessor
  312. * @param manager ProcessingManager that is in charge of this IRCProcessor
  313. */
  314. protected ProcessMode (IRCParser parser, ProcessingManager manager) { super(parser, manager); }
  315. }