| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798 |
- // Copyright (C) 2016 The Syncthing Authors.
- //
- // This Source Code Form is subject to the terms of the Mozilla Public
- // License, v. 2.0. If a copy of the MPL was not distributed with this file,
- // You can obtain one at http://mozilla.org/MPL/2.0/.
- // Adapted from https://github.com/jackpal/Taipei-Torrent/blob/dd88a8bfac6431c01d959ce3c745e74b8a911793/IGD.go
- // Copyright (c) 2010 Jack Palevich (https://github.com/jackpal/Taipei-Torrent/blob/dd88a8bfac6431c01d959ce3c745e74b8a911793/LICENSE)
- package upnp
- import (
- "encoding/xml"
- "fmt"
- "net"
- "time"
- "github.com/syncthing/syncthing/lib/nat"
- )
- // An IGDService is a specific service provided by an IGD.
- type IGDService struct {
- ID string
- URL string
- URN string
- }
- // AddPortMapping adds a port mapping to the specified IGD service.
- func (s *IGDService) AddPortMapping(localIPAddress net.IP, protocol nat.Protocol, internalPort, externalPort int, description string, duration time.Duration) error {
- tpl := `<u:AddPortMapping xmlns:u="%s">
- <NewRemoteHost></NewRemoteHost>
- <NewExternalPort>%d</NewExternalPort>
- <NewProtocol>%s</NewProtocol>
- <NewInternalPort>%d</NewInternalPort>
- <NewInternalClient>%s</NewInternalClient>
- <NewEnabled>1</NewEnabled>
- <NewPortMappingDescription>%s</NewPortMappingDescription>
- <NewLeaseDuration>%d</NewLeaseDuration>
- </u:AddPortMapping>`
- body := fmt.Sprintf(tpl, s.URN, externalPort, protocol, internalPort, localIPAddress, description, duration/time.Second)
- response, err := soapRequest(s.URL, s.URN, "AddPortMapping", body)
- if err != nil && duration > 0 {
- // Try to repair error code 725 - OnlyPermanentLeasesSupported
- envelope := &soapErrorResponse{}
- if unmarshalErr := xml.Unmarshal(response, envelope); unmarshalErr != nil {
- return unmarshalErr
- }
- if envelope.ErrorCode == 725 {
- return s.AddPortMapping(localIPAddress, protocol, externalPort, internalPort, description, 0)
- }
- }
- return err
- }
- // DeletePortMapping deletes a port mapping from the specified IGD service.
- func (s *IGDService) DeletePortMapping(protocol nat.Protocol, externalPort int) error {
- tpl := `<u:DeletePortMapping xmlns:u="%s">
- <NewRemoteHost></NewRemoteHost>
- <NewExternalPort>%d</NewExternalPort>
- <NewProtocol>%s</NewProtocol>
- </u:DeletePortMapping>`
- body := fmt.Sprintf(tpl, s.URN, externalPort, protocol)
- _, err := soapRequest(s.URL, s.URN, "DeletePortMapping", body)
- if err != nil {
- return err
- }
- return nil
- }
- // GetExternalIPAddress queries the IGD service for its external IP address.
- // Returns nil if the external IP address is invalid or undefined, along with
- // any relevant errors
- func (s *IGDService) GetExternalIPAddress() (net.IP, error) {
- tpl := `<u:GetExternalIPAddress xmlns:u="%s" />`
- body := fmt.Sprintf(tpl, s.URN)
- response, err := soapRequest(s.URL, s.URN, "GetExternalIPAddress", body)
- if err != nil {
- return nil, err
- }
- envelope := &soapGetExternalIPAddressResponseEnvelope{}
- err = xml.Unmarshal(response, envelope)
- if err != nil {
- return nil, err
- }
- result := net.ParseIP(envelope.Body.GetExternalIPAddressResponse.NewExternalIPAddress)
- return result, nil
- }
|