package manet

import (
	

	ma 
)

// Loopback Addresses
var (
	// IP4Loopback is the ip4 loopback multiaddr
	IP4Loopback = ma.StringCast("/ip4/127.0.0.1")

	// IP6Loopback is the ip6 loopback multiaddr
	IP6Loopback = ma.StringCast("/ip6/::1")

	// IP4MappedIP6Loopback is the IPv4 Mapped IPv6 loopback address.
	IP4MappedIP6Loopback = ma.StringCast("/ip6/::ffff:127.0.0.1")
)

// Unspecified Addresses (used for )
var (
	IP4Unspecified = ma.StringCast("/ip4/0.0.0.0")
	IP6Unspecified = ma.StringCast("/ip6/::")
)

// IsThinWaist returns whether a Multiaddr starts with "Thin Waist" Protocols.
// This means: /{IP4, IP6}[/{TCP, UDP}]
func ( ma.Multiaddr) bool {
	 = zoneless()
	if  == nil {
		return false
	}
	 := .Protocols()

	// nothing? not even a waist.
	if len() == 0 {
		return false
	}

	if [0].Code != ma.P_IP4 && [0].Code != ma.P_IP6 {
		return false
	}

	// only IP? still counts.
	if len() == 1 {
		return true
	}

	switch [1].Code {
	case ma.P_TCP, ma.P_UDP, ma.P_IP4, ma.P_IP6:
		return true
	default:
		return false
	}
}

// IsIPLoopback returns whether a Multiaddr starts with a "Loopback" IP address
// This means either /ip4/127.*.*.*/*, /ip6/::1/*, or /ip6/::ffff:127.*.*.*.*/*,
// or /ip6zone/<any value>/ip6/<one of the preceding ip6 values>/*
func ( ma.Multiaddr) bool {
	 = zoneless()
	if  == nil {
		return false
	}
	,  := ma.SplitFirst()
	if  == nil {
		return false
	}
	switch .Protocol().Code {
	case ma.P_IP4, ma.P_IP6:
		return net.IP(.RawValue()).IsLoopback()
	}
	return false
}

// IsIP6LinkLocal returns whether a Multiaddr starts with an IPv6 link-local
// multiaddress (with zero or one leading zone). These addresses are non
// routable.
func ( ma.Multiaddr) bool {
	 = zoneless()
	if  == nil {
		return false
	}
	,  := ma.SplitFirst()
	if  == nil || .Protocol().Code != ma.P_IP6 {
		return false
	}
	 := net.IP(.RawValue())
	return .IsLinkLocalMulticast() || .IsLinkLocalUnicast()
}

// IsIPUnspecified returns whether a Multiaddr starts with an Unspecified IP address
// This means either /ip4/0.0.0.0/* or /ip6/::/*
func ( ma.Multiaddr) bool {
	 = zoneless()
	if  == nil {
		return false
	}
	,  := ma.SplitFirst()
	return net.IP(.RawValue()).IsUnspecified()
}

// If m matches [zone,ip6,...], return [ip6,...]
// else if m matches [], [zone], or [zone,...], return nil
// else return m
func zoneless( ma.Multiaddr) ma.Multiaddr {
	,  := ma.SplitFirst()
	if  == nil {
		return nil
	}
	if .Protocol().Code == ma.P_IP6ZONE {
		if  == nil {
			return nil
		}
		,  := ma.SplitFirst()
		if .Protocol().Code != ma.P_IP6 {
			return nil
		}
		return 
	} else {
		return 
	}
}

// IsNAT64IPv4ConvertedIPv6Addr returns whether addr is a well-known prefix "64:ff9b::/96" addr
// used for NAT64 Translation. See RFC 6052
func ( ma.Multiaddr) bool {
	,  := ma.SplitFirst()
	return  != nil && .Protocol().Code == ma.P_IP6 &&
		inAddrRange(.RawValue(), nat64)
}