Skip to content

Commit

Permalink
fix(windows-agent): No empty agent.yaml (#881)
Browse files Browse the repository at this point in the history
When the config is empty, the yaml serialization outputs `{}`. We still
added some commentary in the agent.yaml, resulting in a bunch of nothing
that confused cloud-init, because it's enablement is based on the
existence of the files, their contents are only checked later. That
wouldn't be a problem per se, as the WSL datasource would recognize
there's nothing to do and bail out, but that would have costed
unnecesary CPU cicles, and worse than that, the current implementation
of cloud-init makes it log as an error the fact that there is nothing to
do at this point, which results in more noise if the user see the output
of `cloud-init status --long`.

The fix is rather simple: do not write an empty object into agent.yaml.
If the config is empty we delete that file. This way, if there is
nothing from the user nor the agent's perspective, cloud-init will
remain trully disabled.

I updated some tests that relied on the fact that cloud-init would
create the agent.yaml file unconditionally, which is no longer the case.

UDENG-4169
  • Loading branch information
CarlosNihelton authored Aug 28, 2024
2 parents e0d42aa + 48ffbf7 commit 68e6564
Show file tree
Hide file tree
Showing 12 changed files with 134 additions and 34 deletions.
6 changes: 3 additions & 3 deletions gui/packages/ubuntupro/lib/l10n/app_en.arb
Original file line number Diff line number Diff line change
Expand Up @@ -53,17 +53,17 @@
"purchaseStatusServer":"Something went wrong with Microsoft Store. Please try again later.",
"purchaseStatusUnknown": "Unknown error when trying to purchase the subscription. Consider restarting this app.",

"landscapeHeading": "You can manage your Ubuntu instances remotely with {landscapeLink}.\n\nConfigure the connection to Landscape:",
"landscapeHeading": "Configure the connection to {landscapeLink} to manage your Ubuntu WSL instances remotely.",
"@landscapeHeading": {
"placeholders": {
"landscapeLink": {
"type": "String"
}
}
},
"landscapeQuickSetupSaas": "Quick Setup (SaaS)",
"landscapeQuickSetupSaas": "Landscape SaaS configuration",
"landscapeQuickSetupSaasHint": "Register with landscape.canonical.com",
"landscapeQuickSetupSelfHosted": "Quick Setup (Self-hosted)",
"landscapeQuickSetupSelfHosted": "Self-hosted Landscape configuration",
"landscapeQuickSetupSelfHostedHint": "Register with your own Landscape server",
"landscapeFQDNLabel": "Landscape server address",
"landscapeFQDNError": "Invalid URI. Format should be a hostname or IP address.",
Expand Down
22 changes: 11 additions & 11 deletions gui/packages/ubuntupro/lib/pages/landscape/landscape_page.dart
Original file line number Diff line number Diff line change
Expand Up @@ -141,7 +141,7 @@ class LandscapeConfigForm extends StatelessWidget {
final lang = AppLocalizations.of(context);

return Padding(
padding: const EdgeInsets.symmetric(vertical: _kHeight),
padding: const EdgeInsets.symmetric(vertical: 2 * _kHeight),
// The FocusTraversalGroup is necessary to keep the tab navigation order wed expect:
// We ping-pong between the radio buttons and the form fields that belong to the selected radio button,
// by assigning odd NumericFocusOrder() values to the radio buttons (on the left) and even values to the form fields,
Expand All @@ -158,17 +158,17 @@ class LandscapeConfigForm extends StatelessWidget {
Flexible(
child: Column(
children: [
FocusTraversalOrder(
order: const NumericFocusOrder(0),
child: _ConfigTypeRadio(
value: LandscapeConfigType.saas,
title: lang.landscapeQuickSetupSaas,
subtitle: lang.landscapeQuickSetupSaasHint,
groupValue: model.configType,
onChanged:
model.isSaaSSupported ? model.setConfigType : null,
if (model.isSaaSSupported)
FocusTraversalOrder(
order: const NumericFocusOrder(0),
child: _ConfigTypeRadio(
value: LandscapeConfigType.saas,
title: lang.landscapeQuickSetupSaas,
subtitle: lang.landscapeQuickSetupSaasHint,
groupValue: model.configType,
onChanged: model.setConfigType,
),
),
),
FocusTraversalOrder(
order: const NumericFocusOrder(2),
child: _ConfigTypeRadio(
Expand Down
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
import 'package:dart_either/dart_either.dart';
import 'package:flutter/material.dart';
import 'package:flutter/services.dart';
import 'package:flutter_gen/gen_l10n/app_localizations.dart';
import 'package:yaru/yaru.dart';
import '../../core/either_value_notifier.dart';
Expand Down Expand Up @@ -78,6 +79,10 @@ class _ProTokenInputFieldState extends State<ProTokenInputField> {
children: [
Expanded(
child: TextField(
inputFormatters: [
// This ignores all sorts of (Unicode) whitespaces (not only at the ends).
FilteringTextInputFormatter.deny(RegExp(r'\s')),
],
autofocus: false,
controller: _controller,
decoration: InputDecoration(
Expand Down
4 changes: 2 additions & 2 deletions gui/packages/ubuntupro/pubspec.lock
Original file line number Diff line number Diff line change
Expand Up @@ -254,10 +254,10 @@ packages:
dependency: "direct main"
description:
name: file_picker
sha256: "1375f8685ca6f0412effecc2db834757e9d0e3e055468053e563794b0755cdcd"
sha256: "167bb619cdddaa10ef2907609feb8a79c16dfa479d3afaf960f8e223f754bf12"
url: "https://pub.dev"
source: hosted
version: "8.1.1"
version: "8.1.2"
fixnum:
dependency: transitive
description:
Expand Down
2 changes: 1 addition & 1 deletion gui/packages/ubuntupro/pubspec.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -37,7 +37,7 @@ dependencies:
base_x: ^2.0.1
crypto: ^3.0.5
dart_either: ^1.0.0
file_picker: ^8.1.1
file_picker: ^8.1.2
flutter:
sdk: flutter
flutter_localizations:
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -137,6 +137,24 @@ void main() {
final input = tester.firstWidget<TextField>(inputField);
expect(input.decoration!.errorText, isNull);

final button =
tester.firstWidget<ElevatedButton>(find.byType(ElevatedButton));
expect(button.enabled, isTrue);
});
testWidgets('good token with spaces', (tester) async {
await tester.pumpWidget(theApp);
final inputField = find.byType(TextField);

await tester.enterText(
inputField,
// good token plus a bunch of types of white spaces.
' ${tks.good} \u{00A0}\u{2000}\u{2002}\u{202F}\u{205F}\u{3000} ',
);
await tester.pump();

final input = tester.firstWidget<TextField>(inputField);
expect(input.decoration!.errorText, isNull);

final button =
tester.firstWidget<ElevatedButton>(find.byType(ElevatedButton));
expect(button.enabled, isTrue);
Expand Down
37 changes: 31 additions & 6 deletions windows-agent/internal/cloudinit/cloudinit.go
Original file line number Diff line number Diff line change
Expand Up @@ -37,6 +37,12 @@ func New(ctx context.Context, conf Config, publicDir string) (CloudInit, error)
conf: conf,
}

// c.writeAgentData() is no longer guaranteed to create the cloud-init directory,
// so let's check now if we have permission to do so.
if err := os.MkdirAll(c.dataDir, 0700); err != nil {
return CloudInit{}, fmt.Errorf("could not create cloud-init directory: %v", err)
}

if err := c.writeAgentData(); err != nil {
return CloudInit{}, err
}
Expand All @@ -60,6 +66,11 @@ func (c CloudInit) writeAgentData() (err error) {
return err
}

// Nothing to write, we don't want an empty agent.yaml confusing the real cloud-init.
if cloudInit == nil {
return removeFileInDir(c.dataDir, "agent.yaml")
}

err = writeFileInDir(c.dataDir, "agent.yaml", cloudInit)
if err != nil {
return err
Expand All @@ -78,6 +89,15 @@ func (c CloudInit) WriteDistroData(distroName string, cloudInit string) error {
return nil
}

// removeFileInDir attempts to remove the file 'dir/file' if it exists. Missing file is not an error.
func removeFileInDir(dir, file string) error {
err := os.Remove(filepath.Join(dir, file))
if err != nil && !os.IsNotExist(err) {
return err
}
return nil
}

// writeFileInDir:
// 1. Creates the directory if it did not exist.
// 2. Creates the file using the temp-then-move pattern. This avoids read/write races.
Expand Down Expand Up @@ -121,12 +141,6 @@ func (c CloudInit) RemoveDistroData(distroName string) (err error) {
}

func marshalConfig(conf Config) ([]byte, error) {
w := &bytes.Buffer{}

if _, err := fmt.Fprintln(w, "#cloud-config\n# This file was generated automatically and must not be edited"); err != nil {
return nil, fmt.Errorf("could not write #cloud-config stenza and warning message: %v", err)
}

contents := make(map[string]interface{})

if err := ubuntuProModule(conf, contents); err != nil {
Expand All @@ -137,11 +151,22 @@ func marshalConfig(conf Config) ([]byte, error) {
return nil, err
}

// If there is no config to write, then let's not write an empty object with comments to avoid confusing cloud-init.
if len(contents) == 0 {
return nil, nil
}

out, err := yaml.Marshal(contents)
if err != nil {
return nil, fmt.Errorf("could not Marshal user data as a YAML: %v", err)
}

w := &bytes.Buffer{}

if _, err := fmt.Fprintln(w, "#cloud-config\n# This file was generated automatically and must not be edited"); err != nil {
return nil, fmt.Errorf("could not write #cloud-config stenza and warning message: %v", err)
}

Check warning on line 168 in windows-agent/internal/cloudinit/cloudinit.go

View check run for this annotation

Codecov / codecov/patch

windows-agent/internal/cloudinit/cloudinit.go#L167-L168

Added lines #L167 - L168 were not covered by tests

if _, err := w.Write(out); err != nil {
return nil, fmt.Errorf("could not write config body: %v", err)
}
Expand Down
44 changes: 39 additions & 5 deletions windows-agent/internal/cloudinit/cloudinit_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -19,9 +19,13 @@ func TestNew(t *testing.T) {

testCases := map[string]struct {
breakWriteAgentData bool
wantErr bool
emptyConfig bool

wantErr bool
wantNoAgentYaml bool
}{
"Success": {},
"No file if there is no config to write into": {emptyConfig: true, wantNoAgentYaml: true},
"Error when cloud-init agent file cannot be written": {breakWriteAgentData: true, wantErr: true},
}

Expand All @@ -33,7 +37,13 @@ func TestNew(t *testing.T) {

publicDir := t.TempDir()

proToken := "test token"
if tc.emptyConfig {
proToken = ""
}

conf := &mockConfig{
proToken: proToken,
subcriptionErr: tc.breakWriteAgentData,
}

Expand All @@ -47,6 +57,10 @@ func TestNew(t *testing.T) {

// We don't assert on specifics, as they are tested in WriteAgentData tests.
path := filepath.Join(publicDir, ".cloud-init", "agent.yaml")
if tc.wantNoAgentYaml {
require.NoFileExists(t, path, "there should be no agent data file if there is no config to write into")
return
}
require.FileExists(t, path, "agent data file was not created when updating the config")
})
}
Expand Down Expand Up @@ -87,9 +101,12 @@ hostagent_uid = landscapeUID1234
badLandscape bool

// Break writing to file
breakDir bool
breakTempFile bool
breakFile bool
breakDir bool
breakTempFile bool
breakFile bool
breakRemovingFile bool

wantAgentYamlAsDir bool
}{
"Success": {},
"Without hostagent UID": {skipHostAgentUID: true},
Expand All @@ -98,6 +115,7 @@ hostagent_uid = landscapeUID1234
"Without Landscape [client] section": {landscapeNoClientSection: true},
"With empty contents": {skipProToken: true, skipLandscapeConf: true},

"Error to remove existing agent.yaml": {skipProToken: true, skipLandscapeConf: true, breakRemovingFile: true, wantAgentYamlAsDir: true},
"Error obtaining pro token": {breakSubscription: true},
"Error obtaining Landscape config": {breakLandscape: true},
"Error with erroneous Landscape config": {badLandscape: true},
Expand Down Expand Up @@ -164,6 +182,11 @@ hostagent_uid = landscapeUID1234
require.NoError(t, os.WriteFile(dir, nil, 0600), "Setup: could not create file to mess with cloud-init directory")
}

if tc.breakRemovingFile {
_ = os.RemoveAll(path)
require.NoError(t, os.MkdirAll(path, 0750), "Creating the directory that breaks removing agent.yaml should not fail")
require.NoError(t, os.WriteFile(filepath.Join(path, "child.txt"), nil, 0600), "Setup: could not create file to mess with agent.yaml")
}
ci.Update(ctx)

// Assert that the file was updated (success case) or that the old one remains (error case)
Expand All @@ -172,6 +195,17 @@ hostagent_uid = landscapeUID1234
return
}

if tc.wantAgentYamlAsDir {
require.DirExists(t, path, "There should be a directory instead of agent.yaml")
return
}

golden := testutils.Path(t)
if _, err = os.Stat(golden); err != nil && os.IsNotExist(err) {
// golden file doesn't exist
require.NoFileExists(t, path, "There should not be cloud-init agent file without useful contents")
return
}
got, err := os.ReadFile(path)
require.NoError(t, err, "There should be no error reading the cloud-init agent file")

Expand Down Expand Up @@ -237,7 +271,7 @@ data:
require.NoError(t, err, "Setup: cloud-init New should return no errors")

if !tc.noOldData {
require.NoError(t, os.MkdirAll(filepath.Dir(path), 0600), "Setup: could not write old distro data directory")
require.NoError(t, os.MkdirAll(filepath.Dir(path), 0700), "Setup: could not write old distro data directory")
require.NoError(t, os.WriteFile(path, []byte(oldCloudInit), 0600), "Setup: could not write old distro data")
}

Expand Down

This file was deleted.

13 changes: 12 additions & 1 deletion windows-agent/internal/proservices/proservices_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -92,12 +92,16 @@ func TestNew(t *testing.T) {
}
require.NoError(t, err, "New should return no error")

// Those lines are just to trigger the registry watcher callbacks, they don't write anything to the agent.yaml file by the time we check, that's why goldens are empty.
err = reg.WriteValue(k, "LandscapeConfig", "[client]\nuser=JohnDoe", true)
require.NoError(t, err, "Setup: could not write LandscapeConfig to the registry mock")
err = reg.WriteValue(k, "UbuntuProToken", "test-token", false)
require.NoError(t, err, "Setup: could not write UbuntuProToken to the registry mock")

agentYamlPath := filepath.Join(publicDir, ".cloud-init", "agent.yaml")

// Wait for the agent.yaml to be written
require.Eventually(t, checkFileExists(agentYamlPath), 5*time.Second, 200*time.Millisecond, "agent.yaml file should have been created with registry data")

got, err := os.ReadFile(filepath.Join(publicDir, ".cloud-init", "agent.yaml"))
require.NoError(t, err, "Setup: could not read agent.yaml file post test completion")
want := testutils.LoadWithUpdateFromGolden(t, string(got))
Expand All @@ -106,6 +110,13 @@ func TestNew(t *testing.T) {
}
}

func checkFileExists(path string) func() bool {
return func() bool {
s, err := os.Stat(path)
return err == nil && !s.IsDir()
}
}

func TestRegisterGRPCServices(t *testing.T) {
t.Parallel()

Expand Down
Original file line number Diff line number Diff line change
@@ -1,3 +1,8 @@
#cloud-config
# This file was generated automatically and must not be edited
{}
landscape:
client:
tags: wsl
user: JohnDoe
ubuntu_pro:
token: test-token
Original file line number Diff line number Diff line change
@@ -1,3 +1,8 @@
#cloud-config
# This file was generated automatically and must not be edited
{}
landscape:
client:
tags: wsl
user: JohnDoe
ubuntu_pro:
token: test-token

0 comments on commit 68e6564

Please sign in to comment.