// Copyright (c) 2015-2023 Jeevanandam M (jeeva@myjeeva.com), All rights reserved.
// resty source code and usage is governed by a MIT style
// license that can be found in the LICENSE file.

package resty

import (
	
	
	
	
	
)

var (
	// Since v2.8.0
	ErrAutoRedirectDisabled = errors.New("auto redirect is disabled")
)

type (
	// RedirectPolicy to regulate the redirects in the resty client.
	// Objects implementing the RedirectPolicy interface can be registered as
	//
	// Apply function should return nil to continue the redirect journey, otherwise
	// return error to stop the redirect.
	RedirectPolicy interface {
		Apply(req *http.Request, via []*http.Request) error
	}

	// The RedirectPolicyFunc type is an adapter to allow the use of ordinary functions as RedirectPolicy.
	// If f is a function with the appropriate signature, RedirectPolicyFunc(f) is a RedirectPolicy object that calls f.
	RedirectPolicyFunc func(*http.Request, []*http.Request) error
)

// Apply calls f(req, via).
func ( RedirectPolicyFunc) ( *http.Request,  []*http.Request) error {
	return (, )
}

// NoRedirectPolicy is used to disable redirects in the HTTP client
//
//	resty.SetRedirectPolicy(NoRedirectPolicy())
func () RedirectPolicy {
	return RedirectPolicyFunc(func( *http.Request,  []*http.Request) error {
		return ErrAutoRedirectDisabled
	})
}

// FlexibleRedirectPolicy is convenient method to create No of redirect policy for HTTP client.
//
//	resty.SetRedirectPolicy(FlexibleRedirectPolicy(20))
func ( int) RedirectPolicy {
	return RedirectPolicyFunc(func( *http.Request,  []*http.Request) error {
		if len() >=  {
			return fmt.Errorf("stopped after %d redirects", )
		}
		checkHostAndAddHeaders(, [0])
		return nil
	})
}

// DomainCheckRedirectPolicy is convenient method to define domain name redirect rule in resty client.
// Redirect is allowed for only mentioned host in the policy.
//
//	resty.SetRedirectPolicy(DomainCheckRedirectPolicy("host1.com", "host2.org", "host3.net"))
func ( ...string) RedirectPolicy {
	 := make(map[string]bool)
	for ,  := range  {
		[strings.ToLower()] = true
	}

	 := RedirectPolicyFunc(func( *http.Request,  []*http.Request) error {
		if  := [getHostname(.URL.Host)]; ! {
			return errors.New("redirect is not allowed as per DomainCheckRedirectPolicy")
		}

		return nil
	})

	return 
}

//‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾‾
// Package Unexported methods
//_______________________________________________________________________

func getHostname( string) ( string) {
	if strings.Index(, ":") > 0 {
		, _, _ = net.SplitHostPort()
	}
	 = strings.ToLower()
	return
}

// By default Golang will not redirect request headers
// after go throwing various discussion comments from thread
// https://github.com/golang/go/issues/4800
// Resty will add all the headers during a redirect for the same host
func checkHostAndAddHeaders( *http.Request,  *http.Request) {
	 := getHostname(.URL.Host)
	 := getHostname(.URL.Host)
	if strings.EqualFold(, ) {
		for ,  := range .Header {
			.Header[] = 
		}
	} else { // only library User-Agent header is added
		.Header.Set(hdrUserAgentKey, hdrUserAgentValue)
	}
}