-
Notifications
You must be signed in to change notification settings - Fork 57
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Fix bug with replan/start/restart not updating internal config #96
Conversation
@@ -406,7 +406,7 @@ func (s *serviceData) exited(waitErr error) error { | |||
action, onType := getAction(s.config, waitErr == nil) | |||
switch action { | |||
case plan.ActionIgnore: | |||
logger.Debugf("Service %q %s action is %q, transitioning to stopped state", s.config.Name, onType, action) | |||
logger.Noticef("Service %q %s action is %q, transitioning to stopped state", s.config.Name, onType, action) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This is so all three actions (restart, halt, ignore) have a similar log message with these details. This shouldn't happen often (only on service exit) and seems important.
Going to merge this bug fix without further review as well, so we can include it in the next version of Juju. The fix is two lines (the rest is tests and a logging tweak). |
This includes: * Show process's last output in error msg when service exits quickly: canonical/pebble#69 * Fix subtle Service merging issues: canonical/pebble#98 * Fix bug with replan/start/restart not updating internal config: canonical/pebble#96
#13589 This includes: * Show process's last output in error msg when service exits quickly: canonical/pebble#69 * Fix subtle Service merging issues: canonical/pebble#98 * Fix bug with replan/start/restart not updating internal config: canonical/pebble#96
* MVP verify app health after controller/model upgrade Sometimes after a new release users can have issues with deployed applications during an upgrade. Add some CI to verify that existing applications remain healthy after a controller and model upgrade * Remove NewProviderAddressInSpace constructor Instead use the cleaner NewMachineAddress().AsProviderAddress * Remove NewProviderAddress constructor Instead use the cleaner NewMachineAddress().AsProviderAddress * Ignore exit code for kubectl auth check; * Extract "assume" section feature descriptions to standalone file This should make it easier to locate and manage the list of features in the future. * Update show-model feature test to include supported features expectations * Reorder commands run in run_model_metrics. Ensure that ntp/0 is related to focal/0 and ntp/1 is related to juju-qa-test/0 for consistent test results. * Reduce API server metric cardinality The number of API server metric quantile permutations are growing because the rpc.ErrorCode can lead to too many deviations. The fix for now is to reduce this and only compute success or errors. This way we can be sure that the cardinality can be finite. This should reduce the total number of memory used for every API server instance. * Do not force "stable" channel when re-deploying existing local charms When deploying local charms, they do not get associated with a particular channel as this information is not available at deploy time. When deploying a bundle containing local charms multiple times, we should never force the channel for local charms as this prevents Juju from locating the already-existing local charm causing the error described in LP1954933. * Handles permission errors for listing instance profiles. We have introduced some changes into 2.9 where people with very scoped acess keys can't use the new destroy features that require listing instance profiles. * Remove offers when a model is destroyed * Update pubsub library The pubsub library updates to clean up potential runtime errors. Every so often we can end up with a nil message. Which either means we dropped a message or we're attempting to process a message that isn't there. This can lead to issues in people attempting to use the library. Removing the complexity of knowing when to request the next message. It removes the ambiguity of knowing if you have a next message or not. The trick of knowing when next is blocking or not is also gone, as we can use the existing data channel. If that's already full (it's a buffered channel of 1), then just offer a default case in the select to allow the bypassing of the blocking operation. Message pointers in the implementation is gone, you either have a message or you don't. We no longer need to check for empty. And in doing so, the implementation iterates all the pending handlers at once, without checking for data or next each iteration. The interface is exactly the same, just the underlying queue implementation. * Add explicit owner to params when creating an offer * Ensures that we copy the value of APIInfo into a new reference in order to avoid testing races in the raftlease client tests. * Remove stub sentence from add-machine helper * Remove NewProviderAddresses constructors Instead use the cleaner NewMachineAddresses().AsProviderAddresses * Update to latest version of Pebble This includes: * Show process's last output in error msg when service exits quickly: canonical/pebble#69 * Fix subtle Service merging issues: canonical/pebble#98 * Fix bug with replan/start/restart not updating internal config: canonical/pebble#96 * Ensure 'hostname -f' returns juju-assigned hostname on equinix metal nodes This fixes https://bugs.launchpad.net/juju/+bug/1956538 * Make apiserver log file size configurable * Remove the format2 test charm It is unused and has a broken symlink * The interactive version command doesn't need a controller * Remove txn watcher wrench The following removes the txn watcher wrench. It calls too frequently in production applications. This removes it completely, but I'm not adverse to either creating a controller/model config for enabling wrenches (calling wrench.SetEnable method), similar to the setup for logging config. Alternatively, just throttling the existence of the wench every minute or so. Either way, we just need to back off calling this. * Make log size configurable for other use cases * apiserver/apiserver.go (eg: /var/log/juju/logsink.log) * cmd/jujud/agent/machine.go (eg: /var/log/juju/machine-0.log) * worker/deployer/unit_agent.go (eg: /var/log/juju/unit-mysql-0.log) * Update relevant mocks * Fix potential race with signal handling and make code go vet clean This is using: go vet -composites=false ./... * Need to serialize the new logging settings * juju info/find/download run without a controller * Remove application worker from caasapplicationprovisioner runner when it is stopped; * Unit machine test fixes for firewalled env (s390x) The following patches the charmrevision Facade to prevent outbound API requests to external services. This isn't ideal, as I don't really want to patch a function call in a unit tests, but this is an integration masking as a unit test. * Add more properties to ProviderAddress And also add accompanying functional options This is so these params can be included inside addresses, meaning InterfaceInfos doesn't need to duplicate each nic for each address * Actually add new logging settings to controller config * Show the correct config details in "juju bootstrap -h" output * Consistently log "created rotating log file" messages * Add test for app worker shutdown for a dead app and refactor CAASApplicationSuite; * Remove app worker from runner when the worker peacefully exits; * Add support for mgo scram-sha256 authl default to mongo 4.4 on bootstrap * Fixes racy tests for the CAAS firewaller worker. * Softens Raft op queue test to ensure immediate dispatch instead of batches of 1. We have observed that on slow systems, such as our arm64 test node, that there can be dequeued batches with more than one operation. * Prevent leaking of mongo connections The following prevents the leaking of a mongo connection when iterating over the collections. The defer would only be registered for the first one because of the for loop semantics. The fix is simple, wrap it in a closure. * Fix some intermittent unit test failures * Use mongo 4.4 for macos client tests * Run mock gen; * Change to makefile to be consistent with develop branch for mongo dependencies * Increment juju to 2.9.24 * Updates state package for better handling of deferred collection closure. * Updates state package to ensure that we close query iterators upon every usage. * Add missing wiring to instancecfg and worker/deployer * Update import paths to juju/lumberjack repo (instead of go.mod replace) Oddly, go.sum now references gopkg.in/natefinch/lumberjack.v2, but that's because the juju/lumberjack tests reference that import path. Later we should clean up the juju/lumberjack repo, add go.mod there and update the import paths on the tests in that repo. * Fix unmanaged "controller" namespace clash with "controller" model. * go fmt - order imports * Add a couple of wiring tests * Fix intermittent peergrouper worker test failures * Reduce default agent-logfile-max-size from 300MB to 100MB Per Ian's code review comment * Use agent-logfile settings for unit agent (instead of model-logfile) * Remove model-logfile settings from agent wiring * Process offer permissions when migrating models * Remove offer permissions when the offer is removed * Fix TestApplicationsWithExposedOffers unit test * Prevent panic during reporting in relation state tracker. The following prevents the reporting in relation state tracker causing a panic if the relation doesn't exist. Additionally, we may want to check if the relationer is dying before inspecting the relation unit. * Use juju/retry to handle retries - service * Use juju/retry to handle retries - windows * Fixes variable clash and minor comment issues in the instance-poller worker. * Use juju/retry to handle retries - common * Use juju/retry to handle retries - commands * Ensures that the instance-poller API does not attempt to update link-layer device addresses using network config that has no device information. This occurs when providers do not implement the NetworkInterfaces method, and the instance-poller worker supplies "fake" config built from instance addresses. * Fix various tests for agent-logfile changes * Use juju/retry to handle retries - state Turns out in this case the only function in question wasn't in use, so I've droppoed it * Engine Report: Model Cache The following improves the reporting for the model cache. Currently, it's not possible to know what the current state of the model cache is for a given controller. In an ideal world we would have an API to dump the model cache, except that falls down for HA environments. The correct way would be to query every controller in the fleet to report back their current information. That's a bit of work and something that can be done in the future, but for now, we can piggy-back off the juju_engine_report to dump out more information. We don't dump all the information, just enough to help diagnose potential problems with in the eventual consistent cache. * Improve yaml parsing for yaml files without 'clouds:' keyword ... ... so that the 'list-clouds' output can be used directly as an input to 'update-clouds'. * Try to coerce expected format with schema * Style fix camelCase * Retry verify 3 times until success There is a rare race condition where this job will poll for juju status before an upgrade has completed, leading to an error. Resolve this by retrying a small number of times Also return correct error code on deploy apache timeout * Removes bridge-utils from the list of installed packages on Ubuntu. Juju doesn't use it. * Removes the antiquated LXD setup script. No one needs this to get the localhost/lxd provider working any more. * Removes unused testing charms client-forwardproxy and squid-forwardproxy. * Removes some old TODO comments that are no longer relevant. * Fixes nested deployer report test to ensure that all workers are reported as starting, not just the third, which does not guarantee that the others are up. This can fail on slow machines and has been observed on arm64 test jobs. * Improve startUnitRequest to start a new agent if the current one isn't running. Used by juju_start_unit introspection. The current version only restarts the workers of the agent. However if the agent is lost and not running, there is no way to get it back without manual intervention in the machine's agent.conf. Not a user friendly method. * Add core/network helper to normalize MAC addresses * Revert "Merge pull request #13546 from manadart/2.9-unit-charm-url" This reverts commit c5e5548, reversing changes made to 878fc7c. * Ensure MAC addresses are normalized when converting from/to API params This change ensures that the network-related facades always process MAC addresses consistently. * Implement NetworkInterfaces call for azure provider * Record the error if an external cmr controller goes bad * Ensure that public IPs are only reported as shadow addresses when listing NICs This makes the provider's behavior consistent with the rest of the providers. * Bump AllModelWatcher and AllWatcher facade version, remove .Parameters and .Results from the action delta in the new facades; * Add tests for the action delta change; * Re-generate schema; * add message query to juju wait-for * fix linting * Move mockgen to package_test.go; * Use default storage class for AKS CI tests; * ran autolinter * Ensures that unit public address is acquired one time on an as-needed basis rather than pre-populating for every hook context. This should result in fewer API calls, and lower chance of the address not yet being populated. * Stop polling external cmr controller when it is removed * Implement NetworkInterfaces call for openstack provider * Remove fake NIC generation fallback logic from instancepoller worker Now that all Juju providers (except manual) implement the NetworkInterfaces method we can finally remove the fallback logic within the instancepoller worker that would otherwise generate a list of fake NICs to emulate the output of NetworkInterfaces. The prior logic has been replaced with a check that outputs a warning to the logs if the instancepoller ever encounters a provider that does not implement this method. This is meant as a hint to folks working on adding support for new providers in the future that they must implement the method. Note that the instancepoller ignores manual machines so this particular call does not need to be implemented for that provider. * Use juju/retry to handle retries - worker/apicaller * Add per controller and per app limits for downloading resources * Code review fixes * Emit instancepoller error if env does not implement NetworkInterfaces This commit replaces the warning message introduced via the previous commit with an error if the instancepoller detects a (non-manual) substrate that does not implement the NetworkInterfaces method. * Revert "Ensures that the instance-poller API does not attempt to update" This reverts commit fd5be60 which has been made redundant due to the changes in the previous 2 commits. * Bump charmrepo/v6 dependency * Implement GetEssentialMetadata for charmstore repository * Fake GetEssentialMetadata for charmhub repository The proposed method implementation groups together requests based on the macaroons provided (requests without macaroons are lumped together) and attempts to fetch the data in batches (note that macaroons are not currently supported by charmhub but the repository is wired to use them once support becomes available). At the moment, since the charmhub API (via the refresh method) only allows us to access the metadata.yaml contents, the implementation falls back to downloading the charms and populating the essential metadata responses. When the charmhub API rolls out support for fetching the metadata we need (metadata.yaml, manifest.yaml, config.yaml, lxd-profile.yaml), we will amend the implementation accordingly. For the time being nothing is calling this particular repository method so it is safe to land so we can try out the asynchronous charm download flow (behind a feature flag). * Rebuild repository mocks * Bump juju/retry and juju/errors dependencies * Define async-charm-downloads feature flag This flag will enable us to incrementally test the changes around asynchronous charm downloads without breaking the existing deploy logic. * Refactor charm lookup by URL method in state The original implementation of the lookup code returns ErrNotFound for placeholder and/or charms pending to be uploaded. However, the asynchronous charm downloader code will not use the two-phase commit approach of the current code (i.e. prepare the charm doc entry, download charm, update the entry with both the metadata *and* the stored blob path/SHA256) but rather inserts a charm document with the metadata (but no blob details) and the pending-download flag set. To this end, the lookup code has been modified to inspect the controller flag list and allow pending charms to be retrieved if the async charm download flag is set. This will allow the rest of the existing deploy steps to function properly. * Add helper to create new charm docs and populate the charm metadata fields To avoid breaking changes, instead of modifying AddCharm, a new method has been created and AddCharm has been annotated with a TODO comment that the AddCharm method must be replaced once the remaining of the server-side bundle expansion work lands. * Make the revno threshold for the collection watcher configurable Prior to this change, the collect() function which the collection watcher invokes would flag the ID of documents that have a revno <= 0 as not present. As a result the mergeIDs function which is subsequently called to figure out which changes should be emitted by the watcher would ignore those documents. When a new document gets inserted into a collection it gets assigned a revno of 0. As a result, the collection watcher will not report it even if the configured filter function returns true for the document. By making the revno threshold configurable, we can work around this limitation. The default zero-value of the threshold allows us to retain the original watcher behavior and only override as needed to support special use-cases. * Implement watcher for applications that reference charms not yet downloaded * Add escape hatch to AddCharmXXX to trigger an async charm download The escape hatch is behind the async charm downloads feature flag. When enabled, the AddCharm and friends will fetch the relevant charm metadata and persist it as a new charm document with the pending flag set. Currently, the repository implementation do not provide the means to do a batch give-me-the-charm-meta-only request so the call simply emulates this behavior by downloading the charm (as the current AddCharm implementation does) and reseting the calculated SHA256 value (note: the code uses the repository DownloadCharm method directly (instead of the DownloadAndStore abstraction) so no blobs are stored. The server-side-bundle expansion work will (eventually) provide its own dedicated facade (the batch metadata download functionality will be implemented then) thus rendering the methods in the charms facade obsolete. However, the use of the controller flag allows us to test the async charm download functionality *now* without breaking the existing deploy flow. * Support marshaling/unmarshaling of NotYetAvailable apiserver errors The API server will map NotYetAvailable errors to a 409 (conflict) status code. As per the spec (see https://www.w3.org/Protocols/rfc2616/rfc2616-sec10.html#sec10.4.10): "The request could not be completed due to a conflict with the current state of the resource. This code is only allowed in situations where it is expected that the user might be able to resolve the conflict and resubmit the request". * Reply with HTTP 409 (conflict) when trying to get blob data for pending charms The blob streaming code in apiserver/charms.go now checks whether a charm is pending to be downloaded and replies with a 409 status code instead of a 404 to let the client know that the blob data is not yet available but will be in the future and they need to retry the request. * Update blob downloader client to properly handle 409 status codes 409 status codes are mapped to NotYetAvailable errors. * Create and register charmdownloader facade This is a controller facade which allows clients to: - watch applications with pending charms. - trigger (asynchronously) a download for any pending charms. * Add client for the charmdownloader facade * Create charmdownloader worker This worker detects applications that have a pending charm to be downloaded and triggers an asynchronous download. Note that the watcher will only report applications with pending charms iff the async-charm-downloads flag is set. Consequently, we can safely enable the worker without any issues. * Add the charmdownloader worker to the model manifold * Wrap the bundle reader used by the manifest deployer in a retry loop In the async charm download scenario, having a retry loop allows us to retry the download if the charm takes longer to download than the time it takes for a machine to provision. * Only request supported field names when fetching charmstore metadata The charmstore client implementation for the Metadata call uses reflection to extract the list of names from the provided argument and adds them to the query string that gets sent to the charmstore API. So, instead of passing a core/charm.EssentialCharmMetadata value which includes additional fields that trigger BadRequest errors from the store API we need to define an auxilliary structure with the required field names and copy the response into the EssentialCharmMetadata value. * Update async charm d/l code in charms facade to only fetch essential metadata The initial (placeholder) implementation of the queueAsyncCharmDownload method in the charms facade downloaded the charm and extracted the required metadata from it. With this commit, the implementation now uses the GetEssentialMetadata method from the charm repository to grab the minimum set of metadata needed (metadata.yaml, manifest.yaml, config.yaml and lxd-profile.yaml) for a deployment without having to download the full charm and pass those to state.AddCharmMetadata. The remaining bits of metadata (e.g. actions, metrics etc.) are populated once the charm gets asynchronously downloaded and before the charm blob becomes available for agents to download. * Improve application status messages emitted by the charmdownloader facade * Do not fetch lxd-profile.yaml when fetching essential charm metadata We can simply start LXD containers with the default profile, wait for the charmdownloader worker to fetch and populate the rest of the charm metadata (inc. custom lxd profiles) and rely on the instancemutator worker to apply them to the instances. * Update lxdprofile watcher to emit changes when charm metadata changes When the asynchronous charm download flag is enabled, the charms client facade will only populate the essential bits of the charm document which are required to commence deployment (metadata, manifest and config). The full charm blob is then downloaded asynchronously while the machine for the application units are being provisioned and the full metadata is populated once the download completes. Any potential lxd profile defined by the charm will only become known to Juju once the download is complete. We must therefore rely on the instancemutator worker to apply the lxd profile to the relevant containers. In order for the instancemutator to pick up this change, it needs to watch for charm metadata changes, match the charm URLs to the ones referenced by the applications whose units are assigned to a particular machine and then compare the LXD profiles for changes. This commit introduces this extra bit of functionality. * Run go mod tidy * Rebuild facade schema * Fix AddCharm test * Avoid breaking migration because of NYI Migration fails on some providers (e.g. oci) because of some not yet implemented/optional features. This unblocks it by returning nil whenever it sees an nyi. All the other errors are raised as usual. * Fixes comments for various all-watcher and model-cache logic. * Charmhub refresh results are unordered The following updates the charmhub refresh API clients so that all results are correctly ordered. This was written in the documentation of the charmhub client API, but it was not observed when using the client. This can be seen using the charmrevisionupdater expects all results to be in order. If the results are out of order we end up creating place holders with invalid revisions. These are valid: - ch:amd64/focal/mongodb-64: {} - ch:amd64/focal/ubuntu-19: {} After running they become: - ch:amd64/focal/mongodb-19: {} - ch:amd64/focal/mongodb-64: {} - ch:amd64/focal/ubuntu-19: {} - ch:amd64/focal/ubuntu-64: {} * Upgrade steps to remove orphaned charm docs The following removes charm orphaned docs that where left when we run the upgradecharmrevisioner. This ensures that we don't have any dependency on them to a given application. Hopefully this will remove any potential issue in large deployments with charmhub charms. * Additional logging in PasswordValid failure cases. For LP: 1956975, add error logs on why the password validation failed in the error case. Cause of bug currently undetermined. * Ensure azure NICs have their public IP address (if assigned) populated The recently added implementation for NetworkInterfaces did not populate the public IPs (shadow IPs) section of the returned NIC list. As a result, Juju could not figure out a suitable public IP for workloads and therefore fell back to using the private IP instead. * Use ProviderCallContext instead of context.Context for azure API calls * Prevent charm revision calling on startup On very large deployments we do not want the charm revision worker starting up and instantly calling the charmhub/charmstore APIs. In a HA environment this is compounded by the fact that it does it for every controller node. This can happen if you bounce the agents, so jujus under load or in trouble are not helping themselves out here. The solution here is half baked, I've left a TODO to state what the correct solution is, for now we will just jitter the worker for the delay so that not every node is smashing the API at the same time. The real solution is to have a lease for the worker so that it is only responsible for calling the API. Releasing that lease when it's terminated. We can not add ifResponsible flag in the manifold, because if the agent is being (hard) bounced then there is a chance that this never runs. I think for now, this is a happy medium. * Move 2.9.25 upgrade step to 2.9.24 * Update juju/mutex to juju/mutex/v2 * Ensures that upon cache timeout for setting a unit's charm URL, we indicate whether the unit is in the cache, and/or the *current* value for it's charm URL. * Removes the state serving info dependency from the presence manifold. This is not required, because it is implied by the dependency on the central hub. * Removes the ifController wrapper from manifolds already wrapped by ifDatabaseUpgradeComplete - it is already implied, and therefore redundant. * Transfer machine DisplayName through migration includes temporary code that will be removed before the PR lands: - go.mod for external changed dependency (description pkg), and - provider/oci/environ.go for quick fix to allow the PR to run * Transfer display-name over CloudInstance instead of MachineDoc * Update go.mod & go.sum To use the juju/description edge * Revert back temporary AdoptResources with a TODO * Optimize code a bit DisplayName field is not a pointer, nor omitempty, so no need to check creating extra variables * Add unit tests for migrate_export & import Includes a helper function DisplayName() for state.Machine, normally not available since DisplayName is a field of instanceData (i.e. not Machine) * Re-use existing functionality instead of introducing new method Using existing Machine.InstanceNames() method instead of exporting a new one * move names import to correct stanza * Add Leadership. To improve the run time for juju ssh <app>/leader, add a direct call to get the application's leader unit. Bumped the facade revision. * Add version 3 of the sshclient facade to introduce Leader(). * Updating juju/os to support macos Monterey Adds support for compiling Juju on macos Monterey * Update to use juju/utils/v3 * Make the juju snap strictly confined * Alters the isControllerFlagManifold to depend on the state config watcher instead of state. This means that when a state ping fails, controller-gated workers that do not depend on state will not bounce. * Adds comments to machine manifolds for transitive isController dependencies. * Fixes machine manifolds tests for removed transitive state dependencies. * Remove non-passing acceptance tests In order to get the signal back from our CI tests, we have some tests that just never passed. As the people whom wrote them are long gone and the new preferred way is to write using the shell tests we should nuke from orbit. These tests will be added to a document to state that they do need re-writing in the future, but these tests have always been flaky and we REALLY want to improve our signal to noise ratio. * Fix comment about disabled os (#13669) * Add Leader to SSHClient and start using version 3 of facade. Which implements Leader on the apiserver side. Required to make juju ssh <app>/leader faster, rather than getting the data from status. * Allow Leader to be called during upgrades. It executes a small subset of FullStatus, thus is okay. * Improve time to run juju ssh application/leader. Use a specific API call to get the leader unit name, fall back to using FullStatus if the new sshclient facade version is not available. FullStatus can be very slow on large models, juju ssh should not be if possible. Juju debughooks time to ssh will also improved. Moved mockgen lines to package_test.go, per convention, easier to find in one place. Fixed some unhandled error and unused parameter compiler warnings. * Moved mockgen lines to package_test.go, per convention, easier to find in one place. * Document use of the @ symbol when setting a config value in help message * Comment out the snap interfaces till approved * Comment out the snap plugs till approved Co-authored-by: Juju bot <[email protected]> Co-authored-by: Jack Shaw <[email protected]> Co-authored-by: Kelvin Liu <[email protected]> Co-authored-by: Achilleas Anagnostopoulos <[email protected]> Co-authored-by: Heather Lanigan <[email protected]> Co-authored-by: Thomas Miller <[email protected]> Co-authored-by: Ian Booth <[email protected]> Co-authored-by: Joseph Phillips <[email protected]> Co-authored-by: Ben Hoyt <[email protected]> Co-authored-by: Jujubot <[email protected]> Co-authored-by: Harry Pidcock <[email protected]> Co-authored-by: Caner Derici <[email protected]> Co-authored-by: Bas de Bruijne <[email protected]> Co-authored-by: Arnaud Delobelle <[email protected]>
This fixes a bug in Replan not updating the stored
service.config
value due to bad logic I introduced in https://github.com/canonical/pebble/pull/79/files#diff-c2607d6e10ee84455ecce459b40a302a3a9f44197dc7a2205c52c0cc76c8bfaaR370. I've also added a test for this case.It also fixes a similar bug where a manual stop/start (or restart) sequence didn't update the stored service config -- I introduced this bug in the same commit.