Skip to content

Commit

Permalink
feat: Reduce API calls when creating server
Browse files Browse the repository at this point in the history
- add `--hetzner-wait-on-polling` and `--hetzner-wait-for-running-timeout` flags (#107, thanks @inakimalerba for providing a PR)
  • Loading branch information
JonasProgrammer committed Jun 18, 2023
2 parents 5c67918 + 1eb6edd commit 32908bb
Show file tree
Hide file tree
Showing 5 changed files with 66 additions and 36 deletions.
66 changes: 35 additions & 31 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -15,8 +15,8 @@ You can find sources and pre-compiled binaries [here](https://github.com/JonasPr

```bash
# Download the binary (this example downloads the binary for linux amd64)
$ wget https://github.com/JonasProgrammer/docker-machine-driver-hetzner/releases/download/3.13.0/docker-machine-driver-hetzner_3.13.0_linux_amd64.tar.gz
$ tar -xvf docker-machine-driver-hetzner_3.13.0_linux_amd64.tar.gz
$ wget https://github.com/JonasProgrammer/docker-machine-driver-hetzner/releases/download/4.1.0/docker-machine-driver-hetzner_4.1.0_linux_amd64.tar.gz
$ tar -xvf docker-machine-driver-hetzner_4.1.0_linux_amd64.tar.gz

# Make it executable and copy the binary in a directory accessible with your $PATH
$ chmod +x docker-machine-driver-hetzner
Expand Down Expand Up @@ -115,6 +115,8 @@ $ docker-machine create \
- `--hetzner-ssh-port`: Change the default SSH-Port
- `--hetzner-primary-ipv4/6`: Sets an existing primary IP (v4 or v6 respectively) for the server, as documented in [Networking](#networking)
- `--hetzner-wait-on-error`: Amount of seconds to wait on server creation failure (0/no wait by default)
- `--hetzner-wait-on-polling`: Amount of seconds to wait between requests when waiting for some state to change. (Default: 1 second)
- `--hetzner-wait-for-running-timeout`: Max amount of seconds to wait until a machine is running. (Default: 0/no timeout)

#### Image selection

Expand Down Expand Up @@ -145,35 +147,37 @@ was used during creation.

#### Environment variables and default values

| CLI option | Environment variable | Default |
|---------------------------------|-------------------------------|----------------------------|
| **`--hetzner-api-token`** | `HETZNER_API_TOKEN` | |
| `--hetzner-image` | `HETZNER_IMAGE` | `ubuntu-20.04` as fallback |
| `--hetzner-image-arch` | `HETZNER_IMAGE_ARCH` | *(infer from server)* |
| `--hetzner-image-id` | `HETZNER_IMAGE_ID` | |
| `--hetzner-server-type` | `HETZNER_TYPE` | `cx11` |
| `--hetzner-server-location` | `HETZNER_LOCATION` | *(let Hetzner choose)* |
| `--hetzner-existing-key-path` | `HETZNER_EXISTING_KEY_PATH` | *(generate new keypair)* |
| `--hetzner-existing-key-id` | `HETZNER_EXISTING_KEY_ID` | 0 *(upload new key)* |
| `--hetzner-additional-key` | `HETZNER_ADDITIONAL_KEYS` | |
| `--hetzner-user-data` | `HETZNER_USER_DATA` | |
| `--hetzner-user-data-file` | `HETZNER_USER_DATA_FILE` | |
| `--hetzner-networks` | `HETZNER_NETWORKS` | |
| `--hetzner-firewalls` | `HETZNER_FIREWALLS` | |
| `--hetzner-volumes` | `HETZNER_VOLUMES` | |
| `--hetzner-use-private-network` | `HETZNER_USE_PRIVATE_NETWORK` | false |
| `--hetzner-disable-public-ipv4` | `HETZNER_DISABLE_PUBLIC_IPV4` | false |
| `--hetzner-disable-public-ipv6` | `HETZNER_DISABLE_PUBLIC_IPV6` | false |
| `--hetzner-disable-public` | `HETZNER_DISABLE_PUBLIC` | false |
| `--hetzner-server-label` | (inoperative) | `[]` |
| `--hetzner-key-label` | (inoperative) | `[]` |
| `--hetzner-placement-group` | `HETZNER_PLACEMENT_GROUP` | |
| `--hetzner-auto-spread` | `HETZNER_AUTO_SPREAD` | false |
| `--hetzner-ssh-user` | `HETZNER_SSH_USER` | root |
| `--hetzner-ssh-port` | `HETZNER_SSH_PORT` | 22 |
| `--hetzner-primary-ipv4` | `HETZNER_PRIMARY_IPV4` | |
| `--hetzner-primary-ipv6` | `HETZNER_PRIMARY_IPV6` | |
| `--hetzner-wait-on-error` | `HETZNER_WAIT_ON_ERROR` | 0 |
| CLI option | Environment variable | Default |
|--------------------------------------|------------------------------------|----------------------------|
| **`--hetzner-api-token`** | `HETZNER_API_TOKEN` | |
| `--hetzner-image` | `HETZNER_IMAGE` | `ubuntu-20.04` as fallback |
| `--hetzner-image-arch` | `HETZNER_IMAGE_ARCH` | *(infer from server)* |
| `--hetzner-image-id` | `HETZNER_IMAGE_ID` | |
| `--hetzner-server-type` | `HETZNER_TYPE` | `cx11` |
| `--hetzner-server-location` | `HETZNER_LOCATION` | *(let Hetzner choose)* |
| `--hetzner-existing-key-path` | `HETZNER_EXISTING_KEY_PATH` | *(generate new keypair)* |
| `--hetzner-existing-key-id` | `HETZNER_EXISTING_KEY_ID` | 0 *(upload new key)* |
| `--hetzner-additional-key` | `HETZNER_ADDITIONAL_KEYS` | |
| `--hetzner-user-data` | `HETZNER_USER_DATA` | |
| `--hetzner-user-data-file` | `HETZNER_USER_DATA_FILE` | |
| `--hetzner-networks` | `HETZNER_NETWORKS` | |
| `--hetzner-firewalls` | `HETZNER_FIREWALLS` | |
| `--hetzner-volumes` | `HETZNER_VOLUMES` | |
| `--hetzner-use-private-network` | `HETZNER_USE_PRIVATE_NETWORK` | false |
| `--hetzner-disable-public-ipv4` | `HETZNER_DISABLE_PUBLIC_IPV4` | false |
| `--hetzner-disable-public-ipv6` | `HETZNER_DISABLE_PUBLIC_IPV6` | false |
| `--hetzner-disable-public` | `HETZNER_DISABLE_PUBLIC` | false |
| `--hetzner-server-label` | (inoperative) | `[]` |
| `--hetzner-key-label` | (inoperative) | `[]` |
| `--hetzner-placement-group` | `HETZNER_PLACEMENT_GROUP` | |
| `--hetzner-auto-spread` | `HETZNER_AUTO_SPREAD` | false |
| `--hetzner-ssh-user` | `HETZNER_SSH_USER` | root |
| `--hetzner-ssh-port` | `HETZNER_SSH_PORT` | 22 |
| `--hetzner-primary-ipv4` | `HETZNER_PRIMARY_IPV4` | |
| `--hetzner-primary-ipv6` | `HETZNER_PRIMARY_IPV6` | |
| `--hetzner-wait-on-error` | `HETZNER_WAIT_ON_ERROR` | 0 |
| `--hetzner-wait-on-polling` | `HETZNER_WAIT_ON_POLLING` | 1 |
| `--hetzner-wait-for-running-timeout` | `HETZNER_WAIT_FOR_RUNNING_TIMEOUT` | 0 |

#### Networking

Expand Down
24 changes: 22 additions & 2 deletions driver/driver.go
Original file line number Diff line number Diff line change
Expand Up @@ -56,6 +56,8 @@ type Driver struct {
cachedAdditionalKeys []*hcloud.SSHKey

WaitOnError int
WaitOnPolling int
WaitForRunningTimeout int

// internal housekeeping
version string
Expand Down Expand Up @@ -97,8 +99,12 @@ const (
defaultSSHPort = 22
defaultSSHUser = "root"

flagWaitOnError = "hetzner-wait-on-error"
defaultWaitOnError = 0
flagWaitOnError = "hetzner-wait-on-error"
defaultWaitOnError = 0
flagWaitOnPolling = "hetzner-wait-on-polling"
defaultWaitOnPolling = 1
flagWaitForRunningTimeout = "hetzner-wait-for-running-timeout"
defaultWaitForRunningTimeout = 0

legacyFlagUserDataFromFile = "hetzner-user-data-from-file"
legacyFlagDisablePublic4 = "hetzner-disable-public-4"
Expand Down Expand Up @@ -298,6 +304,18 @@ func (d *Driver) GetCreateFlags() []mcnflag.Flag {
Usage: "Wait if an error happens while creating the server",
Value: defaultWaitOnError,
},
mcnflag.IntFlag{
EnvVar: "HETZNER_WAIT_ON_POLLING",
Name: flagWaitOnPolling,
Usage: "Period for waiting between requests when waiting for some state to change",
Value: defaultWaitOnPolling,
},
mcnflag.IntFlag{
EnvVar: "HETZNER_WAIT_FOR_RUNNING_TIMEOUT",
Name: flagWaitForRunningTimeout,
Usage: "Period for waiting for a machine to be running before failing",
Value: defaultWaitForRunningTimeout,
},
}
}

Expand Down Expand Up @@ -339,6 +357,8 @@ func (d *Driver) setConfigFromFlagsImpl(opts drivers.DriverOptions) error {
d.SSHPort = opts.Int(flagSshPort)

d.WaitOnError = opts.Int(flagWaitOnError)
d.WaitOnPolling = opts.Int(flagWaitOnPolling)
d.WaitForRunningTimeout = opts.Int(flagWaitForRunningTimeout)

d.placementGroup = opts.String(flagPlacementGroup)
if opts.Bool(flagAutoSpread) {
Expand Down
2 changes: 1 addition & 1 deletion driver/hetzner_query.go
Original file line number Diff line number Diff line change
Expand Up @@ -144,7 +144,7 @@ func (d *Driver) waitForAction(a *hcloud.Action) error {
return act.Error()
}

time.Sleep(1 * time.Second)
time.Sleep(time.Duration(d.WaitOnPolling) * time.Second)
}
return nil
}
2 changes: 1 addition & 1 deletion driver/networking.go
Original file line number Diff line number Diff line change
Expand Up @@ -93,7 +93,7 @@ func (d *Driver) configureNetworkAccess(srv hcloud.ServerCreateResult) error {
d.IPAddress = server.PrivateNet[0].IP.String()
break
}
time.Sleep(1 * time.Second)
time.Sleep(time.Duration(d.WaitOnPolling) * time.Second)
}
} else if d.DisablePublic4 {
log.Infof("Using public IPv6 network ...")
Expand Down
8 changes: 7 additions & 1 deletion driver/setup.go
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,7 @@ import (
)

func (d *Driver) waitForRunningServer() error {
start_time := time.Now()
for {
srvstate, err := d.GetState()
if err != nil {
Expand All @@ -20,7 +21,12 @@ func (d *Driver) waitForRunningServer() error {
break
}

time.Sleep(1 * time.Second)
elapsed_time := time.Since(start_time).Seconds()
if d.WaitForRunningTimeout > 0 && int(elapsed_time) > d.WaitForRunningTimeout {
return errors.Errorf("server exceeded wait-for-running-timeout.")
}

time.Sleep(time.Duration(d.WaitOnPolling) * time.Second)
}
return nil
}
Expand Down

0 comments on commit 32908bb

Please sign in to comment.