2014-09-05 14:52:55 -04:00
|
|
|
package common
|
|
|
|
|
|
|
|
import (
|
2018-01-22 18:32:33 -05:00
|
|
|
"context"
|
2014-09-05 14:52:55 -04:00
|
|
|
"fmt"
|
2017-01-17 02:19:52 -05:00
|
|
|
"io/ioutil"
|
2014-09-05 14:52:55 -04:00
|
|
|
"log"
|
|
|
|
"math/rand"
|
|
|
|
"net"
|
|
|
|
"net/http"
|
2017-01-17 02:19:52 -05:00
|
|
|
"os"
|
|
|
|
"path/filepath"
|
|
|
|
|
2018-01-19 19:18:44 -05:00
|
|
|
"github.com/hashicorp/packer/helper/multistep"
|
2017-04-04 16:39:01 -04:00
|
|
|
"github.com/hashicorp/packer/packer"
|
2014-09-05 14:52:55 -04:00
|
|
|
)
|
|
|
|
|
|
|
|
// This step creates and runs the HTTP server that is serving files from the
|
|
|
|
// directory specified by the 'http_directory` configuration parameter in the
|
|
|
|
// template.
|
|
|
|
//
|
|
|
|
// Uses:
|
|
|
|
// ui packer.Ui
|
|
|
|
//
|
|
|
|
// Produces:
|
|
|
|
// http_port int - The port the HTTP server started on.
|
|
|
|
type StepHTTPServer struct {
|
|
|
|
HTTPDir string
|
|
|
|
HTTPPortMin uint
|
|
|
|
HTTPPortMax uint
|
|
|
|
|
|
|
|
l net.Listener
|
|
|
|
}
|
|
|
|
|
2018-01-22 18:31:41 -05:00
|
|
|
func (s *StepHTTPServer) Run(_ context.Context, state multistep.StateBag) multistep.StepAction {
|
2014-09-05 14:52:55 -04:00
|
|
|
ui := state.Get("ui").(packer.Ui)
|
|
|
|
|
|
|
|
var httpPort uint = 0
|
|
|
|
if s.HTTPDir == "" {
|
|
|
|
state.Put("http_port", httpPort)
|
|
|
|
return multistep.ActionContinue
|
|
|
|
}
|
|
|
|
|
|
|
|
// Find an available TCP port for our HTTP server
|
|
|
|
var httpAddr string
|
|
|
|
portRange := int(s.HTTPPortMax - s.HTTPPortMin)
|
|
|
|
for {
|
|
|
|
var err error
|
|
|
|
var offset uint = 0
|
|
|
|
|
|
|
|
if portRange > 0 {
|
|
|
|
// Intn will panic if portRange == 0, so we do a check.
|
2017-01-13 17:22:25 -05:00
|
|
|
// Intn is from [0, n), so add 1 to make from [0, n]
|
|
|
|
offset = uint(rand.Intn(portRange + 1))
|
2014-09-05 14:52:55 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
httpPort = offset + s.HTTPPortMin
|
2015-06-08 23:47:47 -04:00
|
|
|
httpAddr = fmt.Sprintf("0.0.0.0:%d", httpPort)
|
2014-09-05 14:52:55 -04:00
|
|
|
log.Printf("Trying port: %d", httpPort)
|
|
|
|
s.l, err = net.Listen("tcp", httpAddr)
|
|
|
|
if err == nil {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ui.Say(fmt.Sprintf("Starting HTTP server on port %d", httpPort))
|
|
|
|
|
|
|
|
// Start the HTTP server and run it in the background
|
|
|
|
fileServer := http.FileServer(http.Dir(s.HTTPDir))
|
|
|
|
server := &http.Server{Addr: httpAddr, Handler: fileServer}
|
|
|
|
go server.Serve(s.l)
|
|
|
|
|
|
|
|
// Save the address into the state so it can be accessed in the future
|
|
|
|
state.Put("http_port", httpPort)
|
2017-01-17 02:19:52 -05:00
|
|
|
SetHTTPPort(fmt.Sprintf("%d", httpPort))
|
2014-09-05 14:52:55 -04:00
|
|
|
|
|
|
|
return multistep.ActionContinue
|
|
|
|
}
|
|
|
|
|
2017-01-17 02:19:52 -05:00
|
|
|
func httpAddrFilename(suffix string) string {
|
|
|
|
uuid := os.Getenv("PACKER_RUN_UUID")
|
|
|
|
return filepath.Join(os.TempDir(), fmt.Sprintf("packer-%s-%s", uuid, suffix))
|
|
|
|
}
|
|
|
|
|
|
|
|
func SetHTTPPort(port string) error {
|
|
|
|
return ioutil.WriteFile(httpAddrFilename("port"), []byte(port), 0644)
|
|
|
|
}
|
|
|
|
|
|
|
|
func SetHTTPIP(ip string) error {
|
|
|
|
return ioutil.WriteFile(httpAddrFilename("ip"), []byte(ip), 0644)
|
|
|
|
}
|
|
|
|
|
|
|
|
func GetHTTPAddr() string {
|
|
|
|
ip, err := ioutil.ReadFile(httpAddrFilename("ip"))
|
|
|
|
if err != nil {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
port, err := ioutil.ReadFile(httpAddrFilename("port"))
|
|
|
|
if err != nil {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
return fmt.Sprintf("%s:%s", ip, port)
|
|
|
|
}
|
|
|
|
|
2014-09-05 14:52:55 -04:00
|
|
|
func (s *StepHTTPServer) Cleanup(multistep.StateBag) {
|
|
|
|
if s.l != nil {
|
|
|
|
// Close the listener so that the HTTP server stops
|
|
|
|
s.l.Close()
|
|
|
|
}
|
2017-01-17 02:19:52 -05:00
|
|
|
os.Remove(httpAddrFilename("port"))
|
|
|
|
os.Remove(httpAddrFilename("ip"))
|
2014-09-05 14:52:55 -04:00
|
|
|
}
|