mirror of
				https://github.com/pion/webrtc.git
				synced 2025-10-26 00:30:35 +08:00 
			
		
		
		
	 1a1488d290
			
		
	
	1a1488d290
	
	
	
		
			
			Expose ToICE() method on ICE candidates to allow for easier conversion to ice package ICE candidates. Resolve https://github.com/pion/webrtc/issues/3069
		
			
				
	
	
		
			244 lines
		
	
	
		
			6.2 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			244 lines
		
	
	
		
			6.2 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // SPDX-FileCopyrightText: 2023 The Pion community <https://pion.ly>
 | |
| // SPDX-License-Identifier: MIT
 | |
| 
 | |
| package webrtc
 | |
| 
 | |
| import (
 | |
| 	"fmt"
 | |
| 
 | |
| 	"github.com/pion/ice/v4"
 | |
| )
 | |
| 
 | |
| // ICECandidate represents a ice candidate.
 | |
| type ICECandidate struct {
 | |
| 	statsID        string
 | |
| 	Foundation     string           `json:"foundation"`
 | |
| 	Priority       uint32           `json:"priority"`
 | |
| 	Address        string           `json:"address"`
 | |
| 	Protocol       ICEProtocol      `json:"protocol"`
 | |
| 	Port           uint16           `json:"port"`
 | |
| 	Typ            ICECandidateType `json:"type"`
 | |
| 	Component      uint16           `json:"component"`
 | |
| 	RelatedAddress string           `json:"relatedAddress"`
 | |
| 	RelatedPort    uint16           `json:"relatedPort"`
 | |
| 	TCPType        string           `json:"tcpType"`
 | |
| 	SDPMid         string           `json:"sdpMid"`
 | |
| 	SDPMLineIndex  uint16           `json:"sdpMLineIndex"`
 | |
| 	extensions     string
 | |
| }
 | |
| 
 | |
| // Conversion for package ice.
 | |
| func newICECandidatesFromICE(
 | |
| 	iceCandidates []ice.Candidate,
 | |
| 	sdpMid string,
 | |
| 	sdpMLineIndex uint16,
 | |
| ) ([]ICECandidate, error) {
 | |
| 	candidates := []ICECandidate{}
 | |
| 
 | |
| 	for _, i := range iceCandidates {
 | |
| 		c, err := newICECandidateFromICE(i, sdpMid, sdpMLineIndex)
 | |
| 		if err != nil {
 | |
| 			return nil, err
 | |
| 		}
 | |
| 		candidates = append(candidates, c)
 | |
| 	}
 | |
| 
 | |
| 	return candidates, nil
 | |
| }
 | |
| 
 | |
| func newICECandidateFromICE(candidate ice.Candidate, sdpMid string, sdpMLineIndex uint16) (ICECandidate, error) {
 | |
| 	typ, err := convertTypeFromICE(candidate.Type())
 | |
| 	if err != nil {
 | |
| 		return ICECandidate{}, err
 | |
| 	}
 | |
| 	protocol, err := NewICEProtocol(candidate.NetworkType().NetworkShort())
 | |
| 	if err != nil {
 | |
| 		return ICECandidate{}, err
 | |
| 	}
 | |
| 
 | |
| 	newCandidate := ICECandidate{
 | |
| 		statsID:       candidate.ID(),
 | |
| 		Foundation:    candidate.Foundation(),
 | |
| 		Priority:      candidate.Priority(),
 | |
| 		Address:       candidate.Address(),
 | |
| 		Protocol:      protocol,
 | |
| 		Port:          uint16(candidate.Port()), //nolint:gosec // G115
 | |
| 		Component:     candidate.Component(),
 | |
| 		Typ:           typ,
 | |
| 		TCPType:       candidate.TCPType().String(),
 | |
| 		SDPMid:        sdpMid,
 | |
| 		SDPMLineIndex: sdpMLineIndex,
 | |
| 	}
 | |
| 
 | |
| 	newCandidate.setExtensions(candidate.Extensions())
 | |
| 
 | |
| 	if candidate.RelatedAddress() != nil {
 | |
| 		newCandidate.RelatedAddress = candidate.RelatedAddress().Address
 | |
| 		newCandidate.RelatedPort = uint16(candidate.RelatedAddress().Port) //nolint:gosec // G115
 | |
| 	}
 | |
| 
 | |
| 	return newCandidate, nil
 | |
| }
 | |
| 
 | |
| // ToICE converts ICECandidate to ice.Candidate.
 | |
| func (c ICECandidate) ToICE() (cand ice.Candidate, err error) {
 | |
| 	candidateID := c.statsID
 | |
| 	switch c.Typ {
 | |
| 	case ICECandidateTypeHost:
 | |
| 		config := ice.CandidateHostConfig{
 | |
| 			CandidateID: candidateID,
 | |
| 			Network:     c.Protocol.String(),
 | |
| 			Address:     c.Address,
 | |
| 			Port:        int(c.Port),
 | |
| 			Component:   c.Component,
 | |
| 			TCPType:     ice.NewTCPType(c.TCPType),
 | |
| 			Foundation:  c.Foundation,
 | |
| 			Priority:    c.Priority,
 | |
| 		}
 | |
| 
 | |
| 		cand, err = ice.NewCandidateHost(&config)
 | |
| 	case ICECandidateTypeSrflx:
 | |
| 		config := ice.CandidateServerReflexiveConfig{
 | |
| 			CandidateID: candidateID,
 | |
| 			Network:     c.Protocol.String(),
 | |
| 			Address:     c.Address,
 | |
| 			Port:        int(c.Port),
 | |
| 			Component:   c.Component,
 | |
| 			Foundation:  c.Foundation,
 | |
| 			Priority:    c.Priority,
 | |
| 			RelAddr:     c.RelatedAddress,
 | |
| 			RelPort:     int(c.RelatedPort),
 | |
| 		}
 | |
| 
 | |
| 		cand, err = ice.NewCandidateServerReflexive(&config)
 | |
| 	case ICECandidateTypePrflx:
 | |
| 		config := ice.CandidatePeerReflexiveConfig{
 | |
| 			CandidateID: candidateID,
 | |
| 			Network:     c.Protocol.String(),
 | |
| 			Address:     c.Address,
 | |
| 			Port:        int(c.Port),
 | |
| 			Component:   c.Component,
 | |
| 			Foundation:  c.Foundation,
 | |
| 			Priority:    c.Priority,
 | |
| 			RelAddr:     c.RelatedAddress,
 | |
| 			RelPort:     int(c.RelatedPort),
 | |
| 		}
 | |
| 
 | |
| 		cand, err = ice.NewCandidatePeerReflexive(&config)
 | |
| 	case ICECandidateTypeRelay:
 | |
| 		config := ice.CandidateRelayConfig{
 | |
| 			CandidateID: candidateID,
 | |
| 			Network:     c.Protocol.String(),
 | |
| 			Address:     c.Address,
 | |
| 			Port:        int(c.Port),
 | |
| 			Component:   c.Component,
 | |
| 			Foundation:  c.Foundation,
 | |
| 			Priority:    c.Priority,
 | |
| 			RelAddr:     c.RelatedAddress,
 | |
| 			RelPort:     int(c.RelatedPort),
 | |
| 		}
 | |
| 
 | |
| 		cand, err = ice.NewCandidateRelay(&config)
 | |
| 	default:
 | |
| 		return nil, fmt.Errorf("%w: %s", errICECandidateTypeUnknown, c.Typ)
 | |
| 	}
 | |
| 
 | |
| 	if cand != nil && err == nil {
 | |
| 		err = c.exportExtensions(cand)
 | |
| 	}
 | |
| 
 | |
| 	return cand, err
 | |
| }
 | |
| 
 | |
| func (c *ICECandidate) setExtensions(ext []ice.CandidateExtension) {
 | |
| 	var extensions string
 | |
| 
 | |
| 	for i := range ext {
 | |
| 		if i > 0 {
 | |
| 			extensions += " "
 | |
| 		}
 | |
| 
 | |
| 		extensions += ext[i].Key + " " + ext[i].Value
 | |
| 	}
 | |
| 
 | |
| 	c.extensions = extensions
 | |
| }
 | |
| 
 | |
| func (c *ICECandidate) exportExtensions(cand ice.Candidate) error {
 | |
| 	extensions := c.extensions
 | |
| 	var ext ice.CandidateExtension
 | |
| 	var field string
 | |
| 
 | |
| 	for i, start := 0, 0; i < len(extensions); i++ {
 | |
| 		switch {
 | |
| 		case extensions[i] == ' ':
 | |
| 			field = extensions[start:i]
 | |
| 			start = i + 1
 | |
| 		case i == len(extensions)-1:
 | |
| 			field = extensions[start:]
 | |
| 		default:
 | |
| 			continue
 | |
| 		}
 | |
| 
 | |
| 		// Extension keys can't be empty
 | |
| 		hasKey := ext.Key != ""
 | |
| 		if !hasKey {
 | |
| 			ext.Key = field
 | |
| 		} else {
 | |
| 			ext.Value = field
 | |
| 		}
 | |
| 
 | |
| 		// Extension value can be empty
 | |
| 		if hasKey || i == len(extensions)-1 {
 | |
| 			if err := cand.AddExtension(ext); err != nil {
 | |
| 				return err
 | |
| 			}
 | |
| 
 | |
| 			ext = ice.CandidateExtension{}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| func convertTypeFromICE(t ice.CandidateType) (ICECandidateType, error) {
 | |
| 	switch t {
 | |
| 	case ice.CandidateTypeHost:
 | |
| 		return ICECandidateTypeHost, nil
 | |
| 	case ice.CandidateTypeServerReflexive:
 | |
| 		return ICECandidateTypeSrflx, nil
 | |
| 	case ice.CandidateTypePeerReflexive:
 | |
| 		return ICECandidateTypePrflx, nil
 | |
| 	case ice.CandidateTypeRelay:
 | |
| 		return ICECandidateTypeRelay, nil
 | |
| 	default:
 | |
| 		return ICECandidateType(t), fmt.Errorf("%w: %s", errICECandidateTypeUnknown, t)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func (c ICECandidate) String() string {
 | |
| 	ic, err := c.ToICE()
 | |
| 	if err != nil {
 | |
| 		return fmt.Sprintf("%#v failed to convert to ICE: %s", c, err)
 | |
| 	}
 | |
| 
 | |
| 	return ic.String()
 | |
| }
 | |
| 
 | |
| // ToJSON returns an ICECandidateInit
 | |
| // as indicated by the spec https://w3c.github.io/webrtc-pc/#dom-rtcicecandidate-tojson
 | |
| func (c ICECandidate) ToJSON() ICECandidateInit {
 | |
| 	candidateStr := ""
 | |
| 
 | |
| 	candidate, err := c.ToICE()
 | |
| 	if err == nil {
 | |
| 		candidateStr = candidate.Marshal()
 | |
| 	}
 | |
| 
 | |
| 	return ICECandidateInit{
 | |
| 		Candidate:     fmt.Sprintf("candidate:%s", candidateStr),
 | |
| 		SDPMid:        &c.SDPMid,
 | |
| 		SDPMLineIndex: &c.SDPMLineIndex,
 | |
| 	}
 | |
| }
 |