2014-09-17 12:53:24 -05:00
|
|
|
package terraform
|
|
|
|
|
2015-01-27 23:48:46 -06:00
|
|
|
import (
|
|
|
|
"strings"
|
|
|
|
)
|
|
|
|
|
2014-10-16 12:04:07 -05:00
|
|
|
// Semaphore is a wrapper around a channel to provide
|
|
|
|
// utility methods to clarify that we are treating the
|
|
|
|
// channel as a semaphore
|
|
|
|
type Semaphore chan struct{}
|
|
|
|
|
|
|
|
// NewSemaphore creates a semaphore that allows up
|
|
|
|
// to a given limit of simultaneous acquisitions
|
|
|
|
func NewSemaphore(n int) Semaphore {
|
|
|
|
if n == 0 {
|
|
|
|
panic("semaphore with limit 0")
|
|
|
|
}
|
|
|
|
ch := make(chan struct{}, n)
|
|
|
|
return Semaphore(ch)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Acquire is used to acquire an available slot.
|
|
|
|
// Blocks until available.
|
|
|
|
func (s Semaphore) Acquire() {
|
|
|
|
s <- struct{}{}
|
|
|
|
}
|
|
|
|
|
|
|
|
// TryAcquire is used to do a non-blocking acquire.
|
|
|
|
// Returns a bool indicating success
|
|
|
|
func (s Semaphore) TryAcquire() bool {
|
|
|
|
select {
|
|
|
|
case s <- struct{}{}:
|
|
|
|
return true
|
|
|
|
default:
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Release is used to return a slot. Acquire must
|
|
|
|
// be called as a pre-condition.
|
|
|
|
func (s Semaphore) Release() {
|
|
|
|
select {
|
|
|
|
case <-s:
|
|
|
|
default:
|
|
|
|
panic("release without an acquire")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-27 23:48:46 -06:00
|
|
|
// resourceProvider returns the provider name for the given type.
|
2015-03-23 17:36:53 -05:00
|
|
|
func resourceProvider(t, alias string) string {
|
|
|
|
if alias != "" {
|
|
|
|
return alias
|
|
|
|
}
|
|
|
|
|
2015-01-27 23:48:46 -06:00
|
|
|
idx := strings.IndexRune(t, '_')
|
|
|
|
if idx == -1 {
|
2016-12-05 11:24:57 -06:00
|
|
|
// If no underscores, the resource name is assumed to be
|
|
|
|
// also the provider name, e.g. if the provider exposes
|
|
|
|
// only a single resource of each type.
|
|
|
|
return t
|
2015-01-27 23:48:46 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
return t[:idx]
|
|
|
|
}
|
|
|
|
|
2014-09-17 12:53:24 -05:00
|
|
|
// strSliceContains checks if a given string is contained in a slice
|
|
|
|
// When anybody asks why Go needs generics, here you go.
|
|
|
|
func strSliceContains(haystack []string, needle string) bool {
|
|
|
|
for _, s := range haystack {
|
|
|
|
if s == needle {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|