Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
P
packer
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Analytics
Analytics
CI / CD
Repository
Value Stream
Wiki
Wiki
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
Kristopher Ruzic
packer
Commits
0dc347c7
Commit
0dc347c7
authored
Aug 30, 2013
by
Mitchell Hashimoto
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
packer: implement Cancel in ProvisionHook
parent
9f559cb2
Changes
5
Show whitespace changes
Inline
Side-by-side
Showing
5 changed files
with
87 additions
and
5 deletions
+87
-5
common/step_provision.go
common/step_provision.go
+1
-0
packer/build.go
packer/build.go
+3
-1
packer/provisioner.go
packer/provisioner.go
+24
-1
packer/provisioner_mock.go
packer/provisioner_mock.go
+8
-1
packer/provisioner_test.go
packer/provisioner_test.go
+51
-2
No files found.
common/step_provision.go
View file @
0dc347c7
...
@@ -42,6 +42,7 @@ func (*StepProvision) Run(state map[string]interface{}) multistep.StepAction {
...
@@ -42,6 +42,7 @@ func (*StepProvision) Run(state map[string]interface{}) multistep.StepAction {
return
multistep
.
ActionContinue
return
multistep
.
ActionContinue
case
<-
time
.
After
(
1
*
time
.
Second
)
:
case
<-
time
.
After
(
1
*
time
.
Second
)
:
if
_
,
ok
:=
state
[
multistep
.
StateCancelled
];
ok
{
if
_
,
ok
:=
state
[
multistep
.
StateCancelled
];
ok
{
log
.
Println
(
"Cancelling provisioning due to interrupt..."
)
hook
.
Cancel
()
hook
.
Cancel
()
return
multistep
.
ActionHalt
return
multistep
.
ActionHalt
}
}
...
...
packer/build.go
View file @
0dc347c7
...
@@ -207,7 +207,9 @@ func (b *coreBuild) Run(originalUi Ui, cache Cache) ([]Artifact, error) {
...
@@ -207,7 +207,9 @@ func (b *coreBuild) Run(originalUi Ui, cache Cache) ([]Artifact, error) {
hooks
[
HookProvision
]
=
make
([]
Hook
,
0
,
1
)
hooks
[
HookProvision
]
=
make
([]
Hook
,
0
,
1
)
}
}
hooks
[
HookProvision
]
=
append
(
hooks
[
HookProvision
],
&
ProvisionHook
{
provisioners
})
hooks
[
HookProvision
]
=
append
(
hooks
[
HookProvision
],
&
ProvisionHook
{
Provisioners
:
provisioners
,
})
}
}
hook
:=
&
DispatchHook
{
Mapping
:
hooks
}
hook
:=
&
DispatchHook
{
Mapping
:
hooks
}
...
...
packer/provisioner.go
View file @
0dc347c7
package
packer
package
packer
import
(
"sync"
)
// A provisioner is responsible for installing and configuring software
// A provisioner is responsible for installing and configuring software
// on a machine prior to building the actual image.
// on a machine prior to building the actual image.
type
Provisioner
interface
{
type
Provisioner
interface
{
...
@@ -25,11 +29,25 @@ type ProvisionHook struct {
...
@@ -25,11 +29,25 @@ type ProvisionHook struct {
// The provisioners to run as part of the hook. These should already
// The provisioners to run as part of the hook. These should already
// be prepared (by calling Prepare) at some earlier stage.
// be prepared (by calling Prepare) at some earlier stage.
Provisioners
[]
Provisioner
Provisioners
[]
Provisioner
lock
sync
.
Mutex
runningProvisioner
Provisioner
}
}
// Runs the provisioners in order.
// Runs the provisioners in order.
func
(
h
*
ProvisionHook
)
Run
(
name
string
,
ui
Ui
,
comm
Communicator
,
data
interface
{})
error
{
func
(
h
*
ProvisionHook
)
Run
(
name
string
,
ui
Ui
,
comm
Communicator
,
data
interface
{})
error
{
defer
func
()
{
h
.
lock
.
Lock
()
defer
h
.
lock
.
Unlock
()
h
.
runningProvisioner
=
nil
}()
for
_
,
p
:=
range
h
.
Provisioners
{
for
_
,
p
:=
range
h
.
Provisioners
{
h
.
lock
.
Lock
()
h
.
runningProvisioner
=
p
h
.
lock
.
Unlock
()
if
err
:=
p
.
Provision
(
ui
,
comm
);
err
!=
nil
{
if
err
:=
p
.
Provision
(
ui
,
comm
);
err
!=
nil
{
return
err
return
err
}
}
...
@@ -40,5 +58,10 @@ func (h *ProvisionHook) Run(name string, ui Ui, comm Communicator, data interfac
...
@@ -40,5 +58,10 @@ func (h *ProvisionHook) Run(name string, ui Ui, comm Communicator, data interfac
// Cancels the privisioners that are still running.
// Cancels the privisioners that are still running.
func
(
h
*
ProvisionHook
)
Cancel
()
{
func
(
h
*
ProvisionHook
)
Cancel
()
{
// TODO(mitchellh): implement
h
.
lock
.
Lock
()
defer
h
.
lock
.
Unlock
()
if
h
.
runningProvisioner
!=
nil
{
h
.
runningProvisioner
.
Cancel
()
}
}
}
packer/provisioner_mock.go
View file @
0dc347c7
...
@@ -3,6 +3,8 @@ package packer
...
@@ -3,6 +3,8 @@ package packer
// MockProvisioner is an implementation of Provisioner that can be
// MockProvisioner is an implementation of Provisioner that can be
// used for tests.
// used for tests.
type
MockProvisioner
struct
{
type
MockProvisioner
struct
{
ProvFunc
func
()
error
PrepCalled
bool
PrepCalled
bool
PrepConfigs
[]
interface
{}
PrepConfigs
[]
interface
{}
ProvCalled
bool
ProvCalled
bool
...
@@ -19,7 +21,12 @@ func (t *MockProvisioner) Prepare(configs ...interface{}) error {
...
@@ -19,7 +21,12 @@ func (t *MockProvisioner) Prepare(configs ...interface{}) error {
func
(
t
*
MockProvisioner
)
Provision
(
ui
Ui
,
comm
Communicator
)
error
{
func
(
t
*
MockProvisioner
)
Provision
(
ui
Ui
,
comm
Communicator
)
error
{
t
.
ProvCalled
=
true
t
.
ProvCalled
=
true
t
.
ProvUi
=
ui
t
.
ProvUi
=
ui
if
t
.
ProvFunc
==
nil
{
return
nil
return
nil
}
return
t
.
ProvFunc
()
}
}
func
(
t
*
MockProvisioner
)
Cancel
()
{
func
(
t
*
MockProvisioner
)
Cancel
()
{
...
...
packer/provisioner_test.go
View file @
0dc347c7
package
packer
package
packer
import
"testing"
import
(
"sync"
"testing"
"time"
)
func
TestProvisionHook_Impl
(
t
*
testing
.
T
)
{
func
TestProvisionHook_Impl
(
t
*
testing
.
T
)
{
var
raw
interface
{}
var
raw
interface
{}
...
@@ -18,7 +22,10 @@ func TestProvisionHook(t *testing.T) {
...
@@ -18,7 +22,10 @@ func TestProvisionHook(t *testing.T) {
var
comm
Communicator
=
nil
var
comm
Communicator
=
nil
var
data
interface
{}
=
nil
var
data
interface
{}
=
nil
hook
:=
&
ProvisionHook
{[]
Provisioner
{
pA
,
pB
}}
hook
:=
&
ProvisionHook
{
Provisioners
:
[]
Provisioner
{
pA
,
pB
},
}
hook
.
Run
(
"foo"
,
ui
,
comm
,
data
)
hook
.
Run
(
"foo"
,
ui
,
comm
,
data
)
if
!
pA
.
ProvCalled
{
if
!
pA
.
ProvCalled
{
...
@@ -30,4 +37,46 @@ func TestProvisionHook(t *testing.T) {
...
@@ -30,4 +37,46 @@ func TestProvisionHook(t *testing.T) {
}
}
}
}
func
TestProvisionHook_cancel
(
t
*
testing
.
T
)
{
var
lock
sync
.
Mutex
order
:=
make
([]
string
,
0
,
2
)
p
:=
&
MockProvisioner
{
ProvFunc
:
func
()
error
{
time
.
Sleep
(
50
*
time
.
Millisecond
)
lock
.
Lock
()
defer
lock
.
Unlock
()
order
=
append
(
order
,
"prov"
)
return
nil
},
}
hook
:=
&
ProvisionHook
{
Provisioners
:
[]
Provisioner
{
p
},
}
finished
:=
make
(
chan
struct
{})
go
func
()
{
hook
.
Run
(
"foo"
,
nil
,
nil
,
nil
)
close
(
finished
)
}()
// Cancel it while it is running
time
.
Sleep
(
10
*
time
.
Millisecond
)
hook
.
Cancel
()
lock
.
Lock
()
order
=
append
(
order
,
"cancel"
)
lock
.
Unlock
()
// Wait
<-
finished
// Verify order
if
order
[
0
]
!=
"cancel"
||
order
[
1
]
!=
"prov"
{
t
.
Fatalf
(
"bad: %#v"
,
order
)
}
}
// TODO(mitchellh): Test that they're run in the proper order
// TODO(mitchellh): Test that they're run in the proper order
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment