multicast_test.go 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168
  1. // Copyright 2012 The Go Authors. All rights reserved.
  2. // Use of this source code is governed by a BSD-style
  3. // license that can be found in the LICENSE file.
  4. // +build darwin freebsd linux netbsd openbsd
  5. package ipv4_test
  6. import (
  7. "code.google.com/p/go.net/ipv4"
  8. "net"
  9. "os"
  10. "testing"
  11. )
  12. func TestReadWriteMulticastIPPayloadUDP(t *testing.T) {
  13. if testing.Short() || !*testExternal {
  14. t.Skip("to avoid external network")
  15. }
  16. c, err := net.ListenPacket("udp4", "224.0.0.0:1024") // see RFC 4727
  17. if err != nil {
  18. t.Fatalf("net.ListenPacket failed: %v", err)
  19. }
  20. defer c.Close()
  21. ifi := loopbackInterface()
  22. if ifi == nil {
  23. t.Skip("an appropriate interface not found")
  24. }
  25. dst, err := net.ResolveUDPAddr("udp4", "224.0.0.254:1024") // see RFC 4727
  26. if err != nil {
  27. t.Fatalf("net.ResolveUDPAddr failed: %v", err)
  28. }
  29. p := ipv4.NewPacketConn(c)
  30. if err := p.JoinGroup(ifi, dst); err != nil {
  31. t.Fatalf("ipv4.PacketConn.JoinGroup on %v failed: %v", ifi, err)
  32. }
  33. if err := p.SetMulticastInterface(ifi); err != nil {
  34. t.Fatalf("ipv4.PacketConn.SetMulticastInterface failed: %v", err)
  35. }
  36. if err := p.SetMulticastLoopback(true); err != nil {
  37. t.Fatalf("ipv4.PacketConn.SetMulticastLoopback failed: %v", err)
  38. }
  39. cf := ipv4.FlagTTL | ipv4.FlagDst | ipv4.FlagInterface
  40. for i, toggle := range []bool{true, false, true} {
  41. if err := p.SetControlMessage(cf, toggle); err != nil {
  42. t.Fatalf("ipv4.PacketConn.SetControlMessage failed: %v", err)
  43. }
  44. writeThenReadPayload(t, i, p, []byte("HELLO-R-U-THERE"), dst)
  45. }
  46. }
  47. func TestReadWriteMulticastIPPayloadICMP(t *testing.T) {
  48. if testing.Short() || !*testExternal {
  49. t.Skip("to avoid external network")
  50. }
  51. if os.Getuid() != 0 {
  52. t.Skip("must be root")
  53. }
  54. c, err := net.ListenPacket("ip4:icmp", "0.0.0.0")
  55. if err != nil {
  56. t.Fatalf("net.ListenPacket failed: %v", err)
  57. }
  58. defer c.Close()
  59. ifi := loopbackInterface()
  60. if ifi == nil {
  61. t.Skip("an appropriate interface not found")
  62. }
  63. dst, err := net.ResolveIPAddr("ip4", "224.0.0.254") // see RFC 4727
  64. if err != nil {
  65. t.Fatalf("net.ResolveIPAddr failed: %v", err)
  66. }
  67. p := ipv4.NewPacketConn(c)
  68. if err := p.JoinGroup(ifi, dst); err != nil {
  69. t.Fatalf("ipv4.PacketConn.JoinGroup on %v failed: %v", ifi, err)
  70. }
  71. if err := p.SetMulticastInterface(ifi); err != nil {
  72. t.Fatalf("ipv4.PacketConn.SetMulticastInterface failed: %v", err)
  73. }
  74. cf := ipv4.FlagTTL | ipv4.FlagDst | ipv4.FlagInterface
  75. for i, toggle := range []bool{true, false, true} {
  76. wb, err := (&icmpMessage{
  77. Type: ipv4.ICMPTypeEcho, Code: 0,
  78. Body: &icmpEcho{
  79. ID: os.Getpid() & 0xffff, Seq: i + 1,
  80. Data: []byte("HELLO-R-U-THERE"),
  81. },
  82. }).Marshal()
  83. if err != nil {
  84. t.Fatalf("icmpMessage.Marshal failed: %v", err)
  85. }
  86. if err := p.SetControlMessage(cf, toggle); err != nil {
  87. t.Fatalf("ipv4.PacketConn.SetControlMessage failed: %v", err)
  88. }
  89. rb := writeThenReadPayload(t, i, p, wb, dst)
  90. m, err := parseICMPMessage(rb)
  91. if err != nil {
  92. t.Fatalf("parseICMPMessage failed: %v", err)
  93. }
  94. if m.Type != ipv4.ICMPTypeEchoReply || m.Code != 0 {
  95. t.Fatalf("got type=%v, code=%v; expected type=%v, code=%v", m.Type, m.Code, ipv4.ICMPTypeEchoReply, 0)
  96. }
  97. }
  98. }
  99. func TestReadWriteMulticastIPDatagram(t *testing.T) {
  100. if testing.Short() || !*testExternal {
  101. t.Skip("to avoid external network")
  102. }
  103. if os.Getuid() != 0 {
  104. t.Skip("must be root")
  105. }
  106. c, err := net.ListenPacket("ip4:icmp", "0.0.0.0")
  107. if err != nil {
  108. t.Fatalf("net.ListenPacket failed: %v", err)
  109. }
  110. defer c.Close()
  111. ifi := loopbackInterface()
  112. if ifi == nil {
  113. t.Skip("an appropriate interface not found")
  114. }
  115. dst, err := net.ResolveIPAddr("ip4", "224.0.0.254") // see RFC 4727
  116. if err != nil {
  117. t.Fatalf("ResolveIPAddr failed: %v", err)
  118. }
  119. r, err := ipv4.NewRawConn(c)
  120. if err != nil {
  121. t.Fatalf("ipv4.NewRawConn failed: %v", err)
  122. }
  123. if err := r.JoinGroup(ifi, dst); err != nil {
  124. t.Fatalf("ipv4.RawConn.JoinGroup on %v failed: %v", ifi, err)
  125. }
  126. if err := r.SetMulticastInterface(ifi); err != nil {
  127. t.Fatalf("ipv4.PacketConn.SetMulticastInterface failed: %v", err)
  128. }
  129. cf := ipv4.FlagTTL | ipv4.FlagDst | ipv4.FlagInterface
  130. for i, toggle := range []bool{true, false, true} {
  131. wb, err := (&icmpMessage{
  132. Type: ipv4.ICMPTypeEcho, Code: 0,
  133. Body: &icmpEcho{
  134. ID: os.Getpid() & 0xffff, Seq: i + 1,
  135. Data: []byte("HELLO-R-U-THERE"),
  136. },
  137. }).Marshal()
  138. if err != nil {
  139. t.Fatalf("icmpMessage.Marshal failed: %v", err)
  140. }
  141. if err := r.SetControlMessage(cf, toggle); err != nil {
  142. t.Fatalf("ipv4.RawConn.SetControlMessage failed: %v", err)
  143. }
  144. rb := writeThenReadDatagram(t, i, r, wb, nil, dst)
  145. m, err := parseICMPMessage(rb)
  146. if err != nil {
  147. t.Fatalf("parseICMPMessage failed: %v", err)
  148. }
  149. if m.Type != ipv4.ICMPTypeEchoReply || m.Code != 0 {
  150. t.Fatalf("got type=%v, code=%v; expected type=%v, code=%v", m.Type, m.Code, ipv4.ICMPTypeEchoReply, 0)
  151. }
  152. }
  153. }