Smack Jingle java.net.BindException: Cannot assign requested address: Cannot bind

解决SmackJingle中NAT穿越问题
本文分析并解决了在使用SmackJingle进行NAT穿越时遇到的异常问题,通过修改候选地址(Candidate)的处理方式,避免了DatagramSocket绑定失败的情况。

运行Smack Jingle demo时(http://fisheye.igniterealtime.org/browse/~raw,r=11613/smack/trunk/jingle/extension/source/org/jivesoftware/smackx/jingle/mediaimpl/demo/Demo.java)总会遇到下面的异常:

Java代码 收藏代码
  1. java.net.BindException:Cannotassignrequestedaddress:Cannotbind
  2. atjava.net.PlainDatagramSocketImpl.bind0(NativeMethod)
  3. atjava.net.PlainDatagramSocketImpl.bind(UnknownSource)
  4. atjava.net.DatagramSocket.bind(UnknownSource)
  5. atjava.net.DatagramSocket.<init>(UnknownSource)
  6. atjava.net.DatagramSocket.<init>(UnknownSource)
  7. atde.javawi.jstun.test.demo.ice.Candidate.<init>(Candidate.java:44)
  8. atde.javawi.jstun.test.demo.ice.ICENegociator.gatherCandidateAddresses(ICENegociator.java:87)
  9. atorg.jivesoftware.smackx.jingle.nat.ICEResolver.initialize(ICEResolver.java:81)
  10. atorg.jivesoftware.smackx.jingle.nat.TransportResolver.initializeAndWait(TransportResolver.java:387)
  11. atorg.jivesoftware.smackx.jingle.nat.ICETransportManager.<init>(ICETransportManager.java:36)
  12. atcom.haojie.smack.demo.Demo.initialize(Demo.java:84)
  13. atcom.haojie.smack.demo.Demo.<init>(Demo.java:76)
  14. atcom.haojie.smack.demo.Demo.main(Demo.java:170)

可以发现是所使用的JSTUN库出了问题. JSTUN位于http://jstun.javawi.de/. 于是下载了jstun-0.7.3.src.tar.gz, 运行其中的de.javawi.jstun.test.demo.ice.ICENegociator, 还是有同样的问题:


Java代码 收藏代码
  1. java.net.BindException:Cannotassignrequestedaddress:Cannotbind
  2. atjava.net.PlainDatagramSocketImpl.bind0(NativeMethod)
  3. atjava.net.PlainDatagramSocketImpl.bind(UnknownSource)
  4. atjava.net.DatagramSocket.bind(UnknownSource)
  5. atjava.net.DatagramSocket.<init>(UnknownSource)
  6. atjava.net.DatagramSocket.<init>(UnknownSource)
  7. atde.javawi.jstun.test.demo.ice.Candidate.<init>(Candidate.java:44)
  8. atde.javawi.jstun.test.demo.ice.ICENegociator.gatherCandidateAddresses(ICENegociator.java:89)
  9. atde.javawi.jstun.test.demo.ice.ICENegociator.main(ICENegociator.java:176)

经过调试,发现了原因.
ICENegociator类的gatherCandidateAddresses()方法是用来收集candidates地址的, 它首先获取local地址,作为host candidate,然后从这个地址向指定的STUN服务器发送STUN请求,获取server reflexive candidate address. 对于这些地址都会创建一个Candidate对象. 在Candidate类的构造函数里问题出现了, 先看Candidate构造函数:


Java代码 收藏代码
  1. publicCandidate(Addressaddress,shortcomponentId)throwsSocketException,UnknownHostException,UtilityException{
  2. this.socket=newDatagramSocket(0,address.getInetAddress());
  3. this.type=CandidateType.Local;
  4. this.componentId=componentId;
  5. this.priority=0;
  6. this.base=this;
  7. this.isInUse=false;
  8. }
  9. publicCandidate(Addressaddress,CandidateTypetype,shortcomponentId,Candidatebase)throwsSocketException,UnknownHostException,UtilityException{
  10. this.socket=newDatagramSocket(0,address.getInetAddress());
  11. this.type=type;
  12. setComponentId(componentId);
  13. this.priority=0;
  14. this.base=base;
  15. this.isInUse=false;
  16. }

在Candidate类的构造函数里,会创建一个DatagramSocket, DatagramSocket构造函数的第二参数是socket要绑定的local address. 对于host candidate, DatagramSocket的创建没有问题,因为host candidate就是local address; 但对于server reflexive candidate address, 问题出现了, 因为这个地址是NAT转换后的地址, socket无法绑定上去的.

所以改动如下, 判断CandidateType, 如果是ServerReflexive, 则不创建socket.主要的代码为:


Java代码 收藏代码
  1. publicclassDiscoveryInfo{
  2. .......
  3. privateintpublicPort;//用于保存NAT转换后的端口
  4. publicintgetPublicPort(){
  5. returnpublicPort;
  6. }
  7. publicvoidsetPublicPort(intpublicPort){
  8. this.publicPort=publicPort;
  9. }
  10. ......
  11. }
  12. publicclassICENegociator{
  13. .......
  14. publicvoidgatherCandidateAddresses(){
  15. ......
  16. DiscoveryTesttest=newDiscoveryTest(iaddress,stunServer,stunPort);
  17. DiscoveryInfodi=test.test();
  18. if(di.getPublicIP()!=null){
  19. Candidatecand=newCandidate(newAddress(di.getPublicIP().getAddress()),CandidateType.ServerReflexive,componentId,local);
  20. cand.setPort(di.getPublicPort());
  21. ......
  22. }
  23. }
  24. ......
  25. }
  26. publicclassDiscoveryTest{
  27. privatebooleantest1()throwsUtilityException,SocketException,UnknownHostException,IOException,MessageAttributeParsingException,MessageHeaderParsingException{
  28. ......
  29. di.setPublicIP(ma.getAddress().getInetAddress());
  30. di.setPublicPort(ma.getPort());
  31. ......
  32. }
  33. }
  34. publicclassCandidateimplementsComparable{
  35. ......
  36. privateintport;
  37. publicCandidate(Addressaddress,CandidateTypetype,shortcomponentId,Candidatebase)throwsSocketException,UnknownHostException,UtilityException{
  38. if(type==CandidateType.Local){
  39. this.socket=newDatagramSocket(0,address.getInetAddress());
  40. this.address=null;
  41. }else{
  42. this.address=address;
  43. this.socket=null;
  44. }
  45. this.type=type;
  46. setComponentId(componentId);
  47. this.priority=0;
  48. this.base=base;
  49. this.isInUse=false;
  50. }
  51. publicAddressgetAddress()throwsUtilityException{
  52. if(type==CandidateType.Local){
  53. returnnewAddress(socket.getLocalAddress().getAddress());
  54. }else{
  55. returnthis.address;
  56. }
  57. }
  58. publicintgetPort(){
  59. if(type==CandidateType.Local){
  60. returnsocket.getLocalPort();
  61. }else{
  62. returnthis.port;
  63. }
  64. }
  65. publicvoidsetPort(intport){
  66. this.port=port;
  67. }
  68. }


转自:http://mysuperbaby.iteye.com/blog/901370

[openfire@yfw Spark]$ xhost +si:localuser:openfire xhost: unable to open display ":0" [openfire@yfw Spark]$ export JAVA_HOME=/usr/lib/jvm/java-11-openjdk [openfire@yfw Spark]$ export PATH=$JAVA_HOME/bin:$PATH [openfire@yfw Spark]$ [openfire@yfw Spark]$ cd /opt/openfire/enterprise/spark/Spark [openfire@yfw Spark]$ [openfire@yfw Spark]$ java \ > -Djava.awt.headless=true \ > -cp "lib/*" \ > org.jivesoftware.launcher.Startup Warning: no working directory set. This might cause updated data to be missed. Please set a system property 'appdir' to the location where Spark is installed to correct this. java.lang.reflect.InvocationTargetException at java.desktop/java.awt.EventQueue.invokeAndWait(EventQueue.java:1367) at java.desktop/java.awt.EventQueue.invokeAndWait(EventQueue.java:1342) at org.jivesoftware.Spark.startup(Spark.java:186) at java.base/jdk.internal.reflect.NativeMethodAccessorImpl.invoke0(Native Method) at java.base/jdk.internal.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62) at java.base/jdk.internal.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43) at java.base/java.lang.reflect.Method.invoke(Method.java:566) at org.jivesoftware.launcher.Startup.start(Startup.java:75) at org.jivesoftware.launcher.Startup.main(Startup.java:31) Caused by: java.awt.HeadlessException at java.desktop/java.awt.GraphicsEnvironment.checkHeadless(GraphicsEnvironment.java:208) at java.desktop/java.awt.Window.<init>(Window.java:548) at java.desktop/java.awt.Frame.<init>(Frame.java:423) at java.desktop/java.awt.Frame.<init>(Frame.java:388) at java.desktop/javax.swing.JFrame.<init>(JFrame.java:180) at org.jivesoftware.Spark.lambda$startup$0(Spark.java:188) at java.desktop/java.awt.event.InvocationEvent.dispatch(InvocationEvent.java:303) at java.desktop/java.awt.EventQueue.dispatchEventImpl(EventQueue.java:770) at java.desktop/java.awt.EventQueue$4.run(EventQueue.java:721) at java.desktop/java.awt.EventQueue$4.run(EventQueue.java:715) at java.base/java.security.AccessController.doPrivileged(Native Method) at java.base/java.security.ProtectionDomain$JavaSecurityAccessImpl.doIntersectionPrivilege(ProtectionDomain.java:85) at java.desktop/java.awt.EventQueue.dispatchEvent(EventQueue.java:740) at java.desktop/java.awt.EventDispatchThread.pumpOneEventForFilters(EventDispatchThread.java:203) at java.desktop/java.awt.EventDispatchThread.pumpEventsForFilter(EventDispatchThread.java:124) at java.desktop/java.awt.EventDispatchThread.pumpEventsForHierarchy(EventDispatchThread.java:113) at java.desktop/java.awt.EventDispatchThread.pumpEvents(EventDispatchThread.java:109) at java.desktop/java.awt.EventDispatchThread.pumpEvents(EventDispatchThread.java:101) at java.desktop/java.awt.EventDispatchThread.run(EventDispatchThread.java:90) [openfire@yfw Spark]$
10-31
[root@yfw ~]# cd /opt/openfire/enterprise/spark/Spark [root@yfw Spark]# # 推荐使用 openfire 用户身份运行(更安全) [root@yfw Spark]# su - openfire -s /bin/bash ✅ JAVA_HOME: /usr/lib/jvm/java-11-openjdk ✅ Maven home: /opt/maven ✅ Java version: openjdk version "11.0.13" 2021-10-19 LTS [openfire@yfw ~]$ [openfire@yfw ~]$ # 切换到 Spark 目录 [openfire@yfw ~]$ cd /opt/openfire/enterprise/spark/Spark [openfire@yfw Spark]$ [openfire@yfw Spark]$ # 设置 DISPLAY 环境变量(通常为 :0) [openfire@yfw Spark]$ export DISPLAY=:0 [openfire@yfw Spark]$ [openfire@yfw Spark]$ # 启动 Spark [openfire@yfw Spark]$ ./Spark ./Spark: line 384: basename: command not found ./Spark: line 385: dirname: command not found ./Spark: line 400: dirname: command not found ./Spark: line 401: basename: command not found ./Spark: line 81: expr: command not found ./Spark: line 82: expr: command not found ./Spark: line 96: awk: command not found ./Spark: line 97: cp: command not found ./Spark: line 100: sed: command not found ./Spark: line 103: chmod: command not found ./Spark: line 172: expr: command not found ./Spark: line 172: [: -eq: unary operator expected ./Spark: line 172: expr: command not found ./Spark: line 172: [: -eq: unary operator expected ./Spark: line 172: expr: command not found ./Spark: line 172: [: -eq: unary operator expected ./Spark: line 172: expr: command not found ./Spark: line 172: [: -eq: unary operator expected ./Spark: line 172: expr: command not found ./Spark: line 172: [: -eq: unary operator expected Error: Could not find or load main class install4j.org.jivesoftware.launcher.Startup [openfire@yfw Spark]$ echo $PATH /usr/local/ffmpeg/bin:/opt/spark/bin:=en_US.UTF-8:/usr/local/ffmpeg/bin [openfire@yfw Spark]$ export PATH="/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin" [openfire@yfw Spark]$ ls /usr/bin | grep -E "(cp|sed|basename|dirname)" basename cp cpack cpack3 cpan cpan-mirrors cpio cpp cpupower dirname lscpu mcpp rpm2cpio scp sed sedismod sedispol tcptraceroute [openfire@yfw Spark]$ which cp # 应该输出 /usr/bin/cp /usr/bin/cp [openfire@yfw Spark]$ which sed # /usr/bin/sed /usr/bin/sed [openfire@yfw Spark]$ which java # 可能是 /usr/lib/jvm/java-11-openjdk/bin/java /usr/bin/java [openfire@yfw Spark]$ java -version # 查看 Java 版本 openjdk version "1.8.0_312" OpenJDK Runtime Environment (build 1.8.0_312-b07) OpenJDK 64-Bit Server VM (build 25.312-b07, mixed mode) [openfire@yfw Spark]$ ls /usr/lib/jvm/java-11-openjdk/ bin conf include legal lib release tapset [openfire@yfw Spark]$ /usr/lib/jvm/java-11-openjdk/bin/java -version openjdk version "11.0.13" 2021-10-19 LTS OpenJDK Runtime Environment 18.9 (build 11.0.13+8-LTS) OpenJDK 64-Bit Server VM 18.9 (build 11.0.13+8-LTS, mixed mode, sharing) [openfire@yfw Spark]$ export JAVA_HOME=/usr/lib/jvm/java-11-openjdk [openfire@yfw Spark]$ export PATH=$JAVA_HOME/bin:/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin [openfire@yfw Spark]$ java -version openjdk version "11.0.13" 2021-10-19 LTS OpenJDK Runtime Environment 18.9 (build 11.0.13+8-LTS) OpenJDK 64-Bit Server VM 18.9 (build 11.0.13+8-LTS, mixed mode, sharing) [openfire@yfw Spark]$ cd /opt/openfire/enterprise/spark/Spark [openfire@yfw Spark]$ ./Spark java.lang.reflect.InvocationTargetException at java.base/jdk.internal.reflect.NativeMethodAccessorImpl.invoke0(Native Method) at java.base/jdk.internal.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62) at java.base/jdk.internal.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43) at java.base/java.lang.reflect.Method.invoke(Method.java:566) at org.jivesoftware.launcher.Startup.start(Startup.java:75) at org.jivesoftware.launcher.Startup.main(Startup.java:31) at java.base/jdk.internal.reflect.NativeMethodAccessorImpl.invoke0(Native Method) at java.base/jdk.internal.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62) at java.base/jdk.internal.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43) at java.base/java.lang.reflect.Method.invoke(Method.java:566) at com.exe4j.runtime.LauncherEngine.launch(LauncherEngine.java:84) at com.install4j.runtime.launcher.UnixLauncher.start(UnixLauncher.java:69) at install4j.org.jivesoftware.launcher.Startup.main(Unknown Source) Caused by: java.awt.AWTError: Can't connect to X11 window server using ':0' as the value of the DISPLAY variable. at java.desktop/sun.awt.X11GraphicsEnvironment.initDisplay(Native Method) at java.desktop/sun.awt.X11GraphicsEnvironment$1.run(X11GraphicsEnvironment.java:102) at java.base/java.security.AccessController.doPrivileged(Native Method) at java.desktop/sun.awt.X11GraphicsEnvironment.<clinit>(X11GraphicsEnvironment.java:61) at java.base/java.lang.Class.forName0(Native Method) at java.base/java.lang.Class.forName(Class.java:315) at java.desktop/java.awt.GraphicsEnvironment$LocalGE.createGE(GraphicsEnvironment.java:101) at java.desktop/java.awt.GraphicsEnvironment$LocalGE.<clinit>(GraphicsEnvironment.java:83) at java.desktop/java.awt.GraphicsEnvironment.getLocalGraphicsEnvironment(GraphicsEnvironment.java:129) at java.desktop/sun.awt.X11.XToolkit.<clinit>(XToolkit.java:231) at java.base/java.lang.Class.forName0(Native Method) at java.base/java.lang.Class.forName(Class.java:315) at java.desktop/java.awt.Toolkit$2.run(Toolkit.java:588) at java.desktop/java.awt.Toolkit$2.run(Toolkit.java:583) at java.base/java.security.AccessController.doPrivileged(Native Method) at java.desktop/java.awt.Toolkit.getDefaultToolkit(Toolkit.java:582) at java.desktop/sun.swing.SwingUtilities2.getSystemMnemonicKeyMask(SwingUtilities2.java:2212) at java.desktop/javax.swing.plaf.basic.BasicLookAndFeel.initComponentDefaults(BasicLookAndFeel.java:1096) at java.desktop/javax.swing.plaf.metal.MetalLookAndFeel.initComponentDefaults(MetalLookAndFeel.java:440) at java.desktop/javax.swing.plaf.basic.BasicLookAndFeel.getDefaults(BasicLookAndFeel.java:150) at java.desktop/javax.swing.plaf.metal.MetalLookAndFeel.getDefaults(MetalLookAndFeel.java:1560) at java.desktop/javax.swing.UIManager.setLookAndFeel(UIManager.java:587) at java.desktop/javax.swing.UIManager.setLookAndFeel(UIManager.java:629) at java.desktop/javax.swing.UIManager.initializeDefaultLAF(UIManager.java:1404) at java.desktop/javax.swing.UIManager.initialize(UIManager.java:1517) at java.desktop/javax.swing.UIManager.maybeInitialize(UIManager.java:1483) at java.desktop/javax.swing.UIManager.getInstalledLookAndFeels(UIManager.java:422) at java.desktop/javax.swing.UIManager.installLookAndFeel(UIManager.java:465) at java.desktop/javax.swing.UIManager.installLookAndFeel(UIManager.java:484) at org.jivesoftware.spark.ui.themes.LookAndFeelManager.<clinit>(LookAndFeelManager.java:40) at org.jivesoftware.Spark.startup(Spark.java:163) ... 13 more [openfire@yfw Spark]$ xhost +si:localuser:openfire xhost: unable to open display ":0" [openfire@yfw Spark]$ export JAVA_HOME=/usr/lib/jvm/java-11-openjdk [openfire@yfw Spark]$ export PATH=$JAVA_HOME/bin:$PATH [openfire@yfw Spark]$ [openfire@yfw Spark]$ cd /opt/openfire/enterprise/spark/Spark [openfire@yfw Spark]$ [openfire@yfw Spark]$ java \ > -Djava.awt.headless=true \ > -cp "lib/*" \ > org.jivesoftware.launcher.Startup Warning: no working directory set. This might cause updated data to be missed. Please set a system property 'appdir' to the location where Spark is installed to correct this. java.lang.reflect.InvocationTargetException at java.desktop/java.awt.EventQueue.invokeAndWait(EventQueue.java:1367) at java.desktop/java.awt.EventQueue.invokeAndWait(EventQueue.java:1342) at org.jivesoftware.Spark.startup(Spark.java:186) at java.base/jdk.internal.reflect.NativeMethodAccessorImpl.invoke0(Native Method) at java.base/jdk.internal.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62) at java.base/jdk.internal.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43) at java.base/java.lang.reflect.Method.invoke(Method.java:566) at org.jivesoftware.launcher.Startup.start(Startup.java:75) at org.jivesoftware.launcher.Startup.main(Startup.java:31) Caused by: java.awt.HeadlessException at java.desktop/java.awt.GraphicsEnvironment.checkHeadless(GraphicsEnvironment.java:208) at java.desktop/java.awt.Window.<init>(Window.java:548) at java.desktop/java.awt.Frame.<init>(Frame.java:423) at java.desktop/java.awt.Frame.<init>(Frame.java:388) at java.desktop/javax.swing.JFrame.<init>(JFrame.java:180) at org.jivesoftware.Spark.lambda$startup$0(Spark.java:188) at java.desktop/java.awt.event.InvocationEvent.dispatch(InvocationEvent.java:303) at java.desktop/java.awt.EventQueue.dispatchEventImpl(EventQueue.java:770) at java.desktop/java.awt.EventQueue$4.run(EventQueue.java:721) at java.desktop/java.awt.EventQueue$4.run(EventQueue.java:715) at java.base/java.security.AccessController.doPrivileged(Native Method) at java.base/java.security.ProtectionDomain$JavaSecurityAccessImpl.doIntersectionPrivilege(ProtectionDomain.java:85) at java.desktop/java.awt.EventQueue.dispatchEvent(EventQueue.java:740) at java.desktop/java.awt.EventDispatchThread.pumpOneEventForFilters(EventDispatchThread.java:203) at java.desktop/java.awt.EventDispatchThread.pumpEventsForFilter(EventDispatchThread.java:124) at java.desktop/java.awt.EventDispatchThread.pumpEventsForHierarchy(EventDispatchThread.java:113) at java.desktop/java.awt.EventDispatchThread.pumpEvents(EventDispatchThread.java:109) at java.desktop/java.awt.EventDispatchThread.pumpEvents(EventDispatchThread.java:101) at java.desktop/java.awt.EventDispatchThread.run(EventDispatchThread.java:90) [openfire@yfw Spark]$ sudo yum install xorg-x11-server-Xvfb -y We trust you have received the usual lecture from the local System Administrator. It usually boils down to these three things: #1) Respect the privacy of others. #2) Think before you type. #3) With great power comes great responsibility. [sudo] password for openfire: Sorry, try again. [sudo] password for openfire: Sorry, try again. [sudo] password for openfire:
10-31
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值