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.

SIPmodel.java 14KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417
  1. /*
  2. * To change this license header, choose License Headers in Project Properties.
  3. * To change this template file, choose Tools | Templates
  4. * and open the template in the editor.
  5. */
  6. package model;
  7. import gov.nist.javax.sip.DialogTimeoutEvent;
  8. import gov.nist.javax.sip.SipListenerExt;
  9. import gui.Hauptfenster;
  10. import java.net.InetAddress;
  11. import java.security.SecureRandom;
  12. import java.text.ParseException;
  13. import java.util.*;
  14. import java.util.logging.Level;
  15. import java.util.logging.Logger;
  16. import javax.sip.DialogTerminatedEvent;
  17. import javax.sip.IOExceptionEvent;
  18. import javax.sip.InvalidArgumentException;
  19. import javax.sip.ListeningPoint;
  20. import javax.sip.RequestEvent;
  21. import javax.sip.ResponseEvent;
  22. import javax.sip.SipException;
  23. import javax.sip.SipFactory;
  24. import javax.sip.SipProvider;
  25. import javax.sip.SipStack;
  26. import javax.sip.TimeoutEvent;
  27. import javax.sip.TransactionTerminatedEvent;
  28. import javax.sip.address.*;
  29. import javax.sip.header.*;
  30. import javax.sip.message.*;
  31. import logger.OhmLogger;
  32. /**
  33. *
  34. * @author Jan komplettes Model für SIPmodel evtl. Vererbung und für jeden SIP
  35. * Response/Request eigene Klasse oder Interface da es langsam unübersichtlich
  36. * wird :( ACK schicken muss noch ergänzt werden BYE muss noch ergänzt werden.
  37. */
  38. public class SIPmodel implements SipListenerExt
  39. {
  40. private static final String TEST = "0123456789abcdefghijklmnopqrstuvwxyz";//für Random String
  41. private static final SecureRandom RANDOM = new SecureRandom(); //für Random String
  42. Hauptfenster view; //später entfernen wegen Zugriff
  43. public static final Logger lg = OhmLogger.getLogger();
  44. public SipFactory sipFactory; // Used to access the SIP API.
  45. public SipStack sipStack; // The SIP stack.
  46. public SipProvider sipProvider; // Used to send SIP messages.
  47. public MessageFactory messageFactory; // Used to create SIP message factory.
  48. public HeaderFactory headerFactory; // Used to create SIP headers.
  49. public AddressFactory addressFactory; // Used to create SIP URIs.
  50. public ListeningPoint listeningPoint; // SIP listening IP address/port.
  51. public Properties properties; // Other properties.
  52. private AudioRinging audioRinging;
  53. private int myPort = 5060;
  54. private String myName = "129";
  55. public String protocolTyp = "UDP";
  56. public String myIPAddress;
  57. public String proxyAddress;
  58. public CallIdHeader callID;
  59. Boolean anrufer; //Hilfsvariable ob ich selbst anrufer bin
  60. Boolean anrufAngenommen;
  61. Boolean anrufCancel;
  62. //Hilfsvariablen wenn ich angerufen werde. evtl. noch optimieren
  63. Boolean werdeAngerufen;
  64. private long cSequenz;
  65. private String proxyTag;
  66. private String myTag;
  67. private String branch;
  68. private String line;
  69. private Address contactAddress;
  70. private ContactHeader contactHeader;
  71. public SIPmodel(Hauptfenster view) //Konstruktor für eigene Daten
  72. {
  73. this.view = view;
  74. anrufer = false;
  75. anrufAngenommen = false;
  76. anrufCancel = false;
  77. proxyAddress = "";
  78. proxyTag = "";
  79. line = "";
  80. cSequenz = 1;
  81. werdeAngerufen = false;
  82. audioRinging = new AudioRinging();
  83. myTag = createString(8); //erzeugt Random Tag mit Länge 8
  84. branch = createString(18);//erzeugt Random branch mit Länge 18
  85. try
  86. {
  87. //wird an sich erstmal nicht gebraucht aber später sinnvoll um eigene Daten zu initialisieren
  88. this.myIPAddress = InetAddress.getLocalHost().getHostAddress();
  89. this.sipFactory = SipFactory.getInstance();
  90. this.sipFactory.setPathName("gov.nist");
  91. this.properties = new Properties();
  92. this.properties.setProperty("javax.sip.STACK_NAME", "stack");
  93. this.sipStack = this.sipFactory.createSipStack(this.properties);
  94. this.messageFactory = this.sipFactory.createMessageFactory();
  95. this.headerFactory = this.sipFactory.createHeaderFactory();
  96. this.addressFactory = this.sipFactory.createAddressFactory();
  97. this.listeningPoint = this.sipStack.createListeningPoint(this.myIPAddress, this.myPort, this.protocolTyp);
  98. this.sipProvider = this.sipStack.createSipProvider(this.listeningPoint);
  99. this.sipProvider.addSipListener(this);
  100. this.contactAddress = this.addressFactory.createAddress("sip:" + this.myName + "@" + this.myIPAddress + ":" + this.myPort);
  101. this.contactHeader = this.headerFactory.createContactHeader(contactAddress);
  102. }
  103. catch (Exception e)
  104. {
  105. System.out.println("Fehler bei Initialisierung eigener Addresse");
  106. System.exit(-1);
  107. }
  108. }
  109. private String createString(int anzahl)
  110. {
  111. StringBuilder sb = new StringBuilder();
  112. for (int i = 0; i < anzahl; i++)
  113. {
  114. sb.append(TEST.charAt(RANDOM.nextInt(TEST.length())));
  115. }
  116. return sb.toString();
  117. }
  118. public void sendRegister(String serverIP, int serverPort) throws ParseException, InvalidArgumentException, SipException
  119. {
  120. proxyAddress = serverIP + ":" + serverPort;
  121. SIPRegister register = new SIPRegister(proxyAddress, myTag, myIPAddress, myPort, branch, addressFactory, headerFactory, messageFactory, contactHeader, sipProvider.getNewCallId());
  122. sipProvider.sendRequest(register.getRequest());
  123. lg.info("Erfolgreiches Senden der Registrierung");
  124. }
  125. public void sendInvitation(String sipaddresse, String proxyaddresse,
  126. int serverPort) throws SipException
  127. { //ServerPort wird nicht verwendet?
  128. resetCallParameter();
  129. anrufer = true;
  130. callID = sipProvider.getNewCallId();
  131. SIPSessionDescription sdp = new SIPSessionDescription(myName, myIPAddress, null, anrufer);
  132. SIPInvite sipinvite = new SIPInvite(sipaddresse, proxyaddresse, myIPAddress, myPort, myTag, branch, callID, cSequenz, addressFactory, sdp.getSessionDescription(), headerFactory, messageFactory, contactHeader);
  133. sipProvider.sendRequest(sipinvite.getInviterequest());
  134. lg.info("Erfolgreiches Senden der Invitation");
  135. }
  136. @Override
  137. public void processRequest(RequestEvent requestEvent)
  138. {
  139. lg.info("Bekomme Anfrage");
  140. if ((Request.OPTIONS).equals(requestEvent.getRequest().getMethod())) //Options Anfrage von Server nach erfolgreicher Registrierung
  141. {
  142. lg.info("Option Request von Server, erstellen 200 OK zu Server");
  143. try
  144. {
  145. SIP200Ok tempOk = new SIP200Ok(messageFactory, requestEvent, contactHeader, headerFactory);
  146. this.sipProvider.sendResponse(tempOk.getResponse());
  147. view.getTxtArea().append("Server Option Request erfolgreich beantwortet\n");
  148. lg.info("Erfolgreiches senden des 200Ok-Headers");
  149. }
  150. catch (SipException ex)
  151. {
  152. lg.getLogger(SIPmodel.class.getName()).log(Level.SEVERE, null, ex);
  153. }
  154. }
  155. //Aktive Kommunikation
  156. else if ((Request.INVITE).equals(requestEvent.getRequest().getMethod())) // Invite von anderem Benutzer
  157. {
  158. resetCallParameter();
  159. anrufer = false;
  160. //Falls Anruf ankommt, wichtige Variablen Zwischenspeichern (Branch, Cseq, CallID, proxyTag)
  161. branch = ((ViaHeader) requestEvent.getRequest().getHeader("Via")).getBranch();
  162. cSequenz = ((CSeqHeader) requestEvent.getRequest().getHeader("Cseq")).getSeqNumber();
  163. proxyTag = ((FromHeader) requestEvent.getRequest().getHeader("From")).getTag();
  164. callID = ((CallIdHeader) requestEvent.getRequest().getHeader("CallID"));
  165. //Standartabfolge sobald man angerufen wird
  166. try
  167. {
  168. //Trying Response erzeugen und abschicken
  169. SIP100Trying tempTrying = new SIP100Trying(messageFactory, requestEvent, headerFactory);
  170. sipProvider.sendResponse(tempTrying.getTryResponse());
  171. view.getTxtArea().append("Trying geschickt\n");
  172. lg.info("Trying ist raus");
  173. //Ringing Response erzeugen und abschicken noch Schleife mit "Anruf annehmen"
  174. SIP180Ringing tempRinging = new SIP180Ringing(requestEvent, myTag, contactHeader, headerFactory, messageFactory);
  175. Timer timer = new Timer();
  176. timer.scheduleAtFixedRate(new TimerTask()
  177. {
  178. @Override
  179. public void run()
  180. {
  181. try
  182. {
  183. if (anrufAngenommen != true) //Falls man nicht abnimmt, klingeln + Ringing schicken
  184. {
  185. sipProvider.sendResponse(tempRinging.getResponseRing());
  186. audioRinging.startRinging();
  187. }
  188. else if(anrufAngenommen==true) //bei annehmen, klingeln aufhören, OK mit SDP schicken
  189. {
  190. audioRinging.endRinging();
  191. SIP200Ok respcall = new SIP200Ok(messageFactory, requestEvent, contactHeader, headerFactory);
  192. SIPSessionDescription sdp = new SIPSessionDescription(myName, myIPAddress, requestEvent, anrufer);
  193. respcall.addSDP(sdp.getSessionDescription(), myTag);
  194. sipProvider.sendResponse(respcall.getResponse());
  195. view.getTxtArea().append("Telefonat beginnt\n");
  196. timer.cancel();
  197. }
  198. }
  199. catch (SipException ex)
  200. {
  201. }
  202. }
  203. }, 0, 1000);
  204. }
  205. catch (SipException ex)
  206. {
  207. lg.getLogger(SIPmodel.class.getName()).log(Level.SEVERE, null, ex);
  208. }
  209. }
  210. else if ((Request.ACK).equals(requestEvent.getRequest().getMethod()))
  211. {
  212. }
  213. else if ((Request.CANCEL).equals(requestEvent.getRequest().getMethod()))
  214. {
  215. view.getTxtArea().append("Anruf wurde gecancelt");
  216. try
  217. {
  218. //Cancel Request -> Response 200 Ok + Response 487 Request Terminated
  219. //200 Ok
  220. anrufCancel = true;
  221. SIP200Ok tempOk = new SIP200Ok(messageFactory, requestEvent, contactHeader, headerFactory);
  222. this.sipProvider.sendResponse(tempOk.getResponse());
  223. view.getTxtArea().append("Anrufender canceld Anfrage\n");
  224. lg.info("Erfolgreiches senden des 200 OkHeaders auf CancelAnfrage");
  225. //487 Request Terminated
  226. }
  227. catch (SipException ex)
  228. {
  229. Logger.getLogger(SIPmodel.class.getName()).log(Level.SEVERE, null, ex);
  230. }
  231. }
  232. else if ((Request.BYE).equals(requestEvent.getRequest().getMethod()))
  233. {
  234. try
  235. {
  236. SIP200Ok tempOk = new SIP200Ok(messageFactory, requestEvent, contactHeader, headerFactory);
  237. this.sipProvider.sendResponse(tempOk.getResponse());
  238. }
  239. catch (SipException ex)
  240. {
  241. Logger.getLogger(SIPmodel.class.getName()).log(Level.SEVERE, null, ex);
  242. }
  243. }
  244. }
  245. @Override
  246. public void processResponse(ResponseEvent responseEvent)
  247. {
  248. if (responseEvent.getResponse().getStatusCode() == Response.UNAUTHORIZED)
  249. {
  250. try
  251. {
  252. cSequenz += 1L;
  253. branch += 10;
  254. SIPUnauthReq req2 = new SIPUnauthReq(myName, proxyAddress, myTag, myIPAddress, branch, cSequenz, myPort, contactHeader, addressFactory, headerFactory, responseEvent, messageFactory);
  255. sipProvider.sendRequest(req2.getUnauthReq());
  256. view.getTxtArea().append("Erfolgreich registriert\n");
  257. }
  258. catch (SipException ex)
  259. {
  260. lg.getLogger(SIPmodel.class.getName()).log(Level.SEVERE, null, ex);
  261. }
  262. }
  263. else if (responseEvent.getResponse().getStatusCode() == Response.OK)
  264. {
  265. lg.info("Erfolgreichen Response (200 OK) bekommen");
  266. }
  267. else if (responseEvent.getResponse().getStatusCode() == Response.RINGING)
  268. {
  269. ToHeader temp = (ToHeader) responseEvent.getResponse().getHeader("To");
  270. proxyTag = temp.getTag();
  271. }
  272. }
  273. public void byecall(String calluser) throws ParseException, InvalidArgumentException, SipException
  274. {
  275. cSequenz++;
  276. SIPBye tempBye = new SIPBye(calluser, proxyAddress, myName, myIPAddress, myTag, proxyTag, myPort, branch, callID, cSequenz, messageFactory, contactHeader, addressFactory, headerFactory);
  277. sipProvider.sendRequest(tempBye.getByeRequest());
  278. lg.info("Erfolgreiches Senden des BYE Headers");
  279. }
  280. public void resetCallParameter()
  281. {
  282. anrufAngenommen = false;
  283. anrufer = false;
  284. }
  285. public void anrufAnnehmen()
  286. {
  287. anrufAngenommen = true;
  288. }
  289. //Get Methoden falls man mal die Parameter braucht
  290. public String getmyName()
  291. {
  292. return myName;
  293. }
  294. public Integer getMyPort()
  295. {
  296. return myPort;
  297. }
  298. public String getMyIPAdress()
  299. {
  300. return myIPAddress;
  301. }
  302. // Herausfinden bei welchen Events die unteren Funktionen aufgerufen werden -> evtl brauchbar
  303. @Override
  304. public void processTimeout(TimeoutEvent timeoutEvent)
  305. {
  306. //view.getTxtArea().append("processTimeout\n");
  307. }
  308. @Override
  309. public void processIOException(IOExceptionEvent exceptionEvent)
  310. {
  311. //view.getTxtArea().append("processIOException\n");
  312. }
  313. @Override
  314. public void processTransactionTerminated(
  315. TransactionTerminatedEvent transactionTerminatedEvent)
  316. {
  317. //view.getTxtArea().append("processTransactionTerminated\n");
  318. }
  319. @Override
  320. public void processDialogTerminated(
  321. DialogTerminatedEvent dialogTerminatedEvent)
  322. {
  323. //view.getTxtArea().append("processDialogTerminated\n");
  324. }
  325. @Override
  326. public void processDialogTimeout(DialogTimeoutEvent timeoutEvent)
  327. {
  328. //view.getTxtArea().append("processDialogTerminated\n");
  329. }
  330. /**
  331. * @param myPort the myPort to set
  332. */
  333. public void setMyPort(int myPort)
  334. {
  335. this.myPort = myPort;
  336. }
  337. /**
  338. * @return the myName
  339. */
  340. public String getMyName()
  341. {
  342. return myName;
  343. }
  344. /**
  345. * @return the proxyTag
  346. */
  347. public String getProxyTag()
  348. {
  349. return proxyTag;
  350. }
  351. /**
  352. * @param proxyTag the proxyTag to set
  353. */
  354. public void setProxyTag(String proxyTag)
  355. {
  356. this.proxyTag = proxyTag;
  357. }
  358. /**
  359. * @return the myTag
  360. */
  361. public String getMyTag()
  362. {
  363. return myTag;
  364. }
  365. /**
  366. * @param myTag the myTag to set
  367. */
  368. public void setMyTag(String myTag)
  369. {
  370. this.myTag = myTag;
  371. }
  372. /**
  373. * @return the branch
  374. */
  375. public String getBranch()
  376. {
  377. return branch;
  378. }
  379. }