Skip to content

Commit a9c262f

Browse files
authored
Merge pull request flyingcircusio#785 from flyingcircusio/PL-131736-enc-release-metadata
Save release ENC metadata, use it in motd
2 parents ede6213 + 00ca550 commit a9c262f

File tree

7 files changed

+89
-14
lines changed

7 files changed

+89
-14
lines changed

nixos/platform/default.nix

+1
Original file line numberDiff line numberDiff line change
@@ -292,6 +292,7 @@ in {
292292
'';
293293

294294
environment.etc."fcio_environment_name".text = config.flyingcircus.enc.parameters.environment or "";
295+
environment.etc."fcio_release".text = config.flyingcircus.platform.release.release_name or "pre";
295296

296297
flyingcircus = {
297298
enc_services = enc_services;

nixos/platform/enc.nix

+20
Original file line numberDiff line numberDiff line change
@@ -64,6 +64,12 @@ with lib;
6464
description = "Where to find the ENC service clients json file.";
6565
};
6666

67+
releasesPath = mkOption {
68+
default = /etc/nixos/releases.json;
69+
defaultText = "/etc/nixos/releases.json";
70+
type = path;
71+
description = "Where to find the releases json file.";
72+
};
6773
systemStatePath = mkOption {
6874
default = /etc/nixos/system_state.json;
6975
type = path;
@@ -84,6 +90,20 @@ with lib;
8490
'';
8591
};
8692

93+
platform = {
94+
release = mkOption {
95+
readOnly = true;
96+
default = (config.flyingcircus.platform.knownReleases.${config.system.nixos.label} or {});
97+
defaultText = "Platform release metadata loaded from the file at `releasesPath`";
98+
};
99+
100+
knownReleases = mkOption {
101+
internal = true;
102+
readOnly = true;
103+
default = fclib.jsonFromFile cfg.releasesPath "{}";
104+
defaultText = "Metadata for all known releases from the file at `releasesPath`";
105+
};
106+
};
87107
};
88108

89109
config = {

nixos/platform/shell.nix

+7-3
Original file line numberDiff line numberDiff line change
@@ -12,6 +12,9 @@ let
1212
in
1313
{
1414
config = {
15+
16+
environment.etc.motd.text = config.users.motd;
17+
1518
environment.interactiveShellInit = ''
1619
export TMOUT=43200
1720
'';
@@ -64,15 +67,16 @@ in
6467
);
6568

6669
users.motd = let
67-
has_release_name = parameters ? release_name;
70+
inherit (config.flyingcircus.platform) release;
71+
isStableRelease = release != {};
6872
in
6973
''
7074
Welcome to the Flying Circus!
7175
7276
Status: https://status.flyingcircus.io/
7377
Docs: https://doc.flyingcircus.io/
74-
${ opt (parameters ? release_changelog ) ("ChangeLog: " + parameters.release_changelog)}
75-
Release: ${ opt has_release_name "${parameters.release_name} (" + config.system.nixos.label + opt has_release_name ")"}
78+
Release: ${opt isStableRelease "${release.release_name} (" + config.system.nixos.label + opt isStableRelease ")"}
79+
${opt isStableRelease ("ChangeLog: " + release.release_changelog)}
7680
7781
'' +
7882
(opt (enc ? name && parameters ? location && parameters ? environment)

pkgs/fc/agent/fc/maintenance/activity/tests/test_update.py

+13-2
Original file line numberDiff line numberDiff line change
@@ -20,6 +20,9 @@
2020
CURRENT_BUILD = 93111
2121
NEXT_BUILD = 93222
2222
NEXT_NEXT_BUILD = 93333
23+
CURRENT_RELEASE = "2021_002"
24+
NEXT_RELEASE = "2021_003"
25+
CHANGELOG_URL = "https://doc.flyingcircus.io/platform/changes/2021/r003.html"
2326
CURRENT_CHANNEL_URL = f"https://hydra.flyingcircus.io/build/{CURRENT_BUILD}/download/1/nixexprs.tar.xz"
2427
NEXT_CHANNEL_URL = f"https://hydra.flyingcircus.io/build/{NEXT_BUILD}/download/1/nixexprs.tar.xz"
2528
ENVIRONMENT = "fc-21.05-production"
@@ -37,7 +40,7 @@
3740
"stop": ["postgresql.service"],
3841
}
3942

40-
CHANGELOG = textwrap.dedent(
43+
SUMMARY = textwrap.dedent(
4144
f"""\
4245
System update: {CURRENT_VERSION} -> {NEXT_VERSION}
4346
@@ -47,21 +50,26 @@
4750
Restart: telegraf
4851
Reload: nginx
4952
53+
Release: {CURRENT_RELEASE} -> {NEXT_RELEASE}
54+
ChangeLog: {CHANGELOG_URL}
5055
Environment: {ENVIRONMENT} (unchanged)
5156
Build number: {CURRENT_BUILD} -> {NEXT_BUILD}
5257
Channel URL: {NEXT_CHANNEL_URL}"""
5358
)
5459

5560
SERIALIZED_ACTIVITY = f"""\
5661
!!python/object:fc.maintenance.activity.update.UpdateActivity
62+
changelog_url: {CHANGELOG_URL}
5763
current_channel_url: https://hydra.flyingcircus.io/build/93111/download/1/nixexprs.tar.xz
5864
current_environment: fc-21.05-production
5965
current_kernel: 5.10.45
66+
current_release: '{CURRENT_RELEASE}'
6067
current_system: {CURRENT_SYSTEM_PATH}
6168
current_version: 21.05.1233.a9cc58d
6269
next_channel_url: https://hydra.flyingcircus.io/build/93222/download/1/nixexprs.tar.xz
6370
next_environment: fc-21.05-production
6471
next_kernel: 5.10.50
72+
next_release: '{NEXT_RELEASE}'
6573
next_system: {NEXT_SYSTEM_PATH}
6674
next_version: 21.05.1235.bacc11d
6775
reboot_needed: !!python/object/apply:fc.maintenance.activity.RebootType
@@ -82,12 +90,15 @@
8290
def activity(logger, nixos_mock):
8391
activity = UpdateActivity(next_channel_url=NEXT_CHANNEL_URL, log=logger)
8492
activity.current_channel_url = CURRENT_CHANNEL_URL
93+
activity.changelog_url = CHANGELOG_URL
8594
activity.current_environment = ENVIRONMENT
95+
activity.current_release = CURRENT_RELEASE
8696
activity.current_version = CURRENT_VERSION
8797
activity.current_kernel = CURRENT_KERNEL_VERSION
8898
activity.next_channel_url = NEXT_CHANNEL_URL
8999
activity.next_environment = ENVIRONMENT
90100
activity.next_kernel = NEXT_KERNEL_VERSION
101+
activity.next_release = NEXT_RELEASE
91102
activity.next_system = NEXT_SYSTEM_PATH
92103
activity.next_version = NEXT_VERSION
93104
activity.reboot_needed = RebootType.WARM
@@ -245,7 +256,7 @@ def test_update_activity_prepare(log, logger, tmp_path, activity, nixos_mock):
245256
assert (
246257
activity.reboot_needed == RebootType.WARM
247258
), "expected warm reboot request"
248-
assert activity.changelog == CHANGELOG
259+
assert activity.summary == SUMMARY
249260

250261
assert log.has(
251262
"update-prepare-start",

pkgs/fc/agent/fc/maintenance/activity/update.py

+12-2
Original file line numberDiff line numberDiff line change
@@ -36,9 +36,12 @@ def __init__(
3636
super().__init__()
3737
self.next_environment = next_environment
3838
self.next_channel_url = next_channel_url
39+
self.changelog_url = None
3940
self.current_system = None
4041
self.next_system = None
4142
self.current_channel_url = None
43+
self.current_release = None
44+
self.next_release = None
4245
self.current_version = None
4346
self.next_version = None
4447
self.current_environment = None
@@ -249,7 +252,7 @@ def run(self):
249252
self.returncode = 0
250253

251254
@property
252-
def changelog(self):
255+
def summary(self):
253256
"""
254257
A human-readable summary of what will be changed by this update.
255258
Includes possible reboots, significant unit state changes (start, stop,
@@ -271,6 +274,13 @@ def changelog(self):
271274
msg.extend(unit_change_lines)
272275
msg.append("")
273276

277+
if self.next_release:
278+
msg.append(
279+
f"Release: {self.current_release} -> {self.next_release}"
280+
)
281+
if self.changelog_url:
282+
msg.append(f"ChangeLog: {self.changelog_url}")
283+
274284
if self.current_environment != self.next_environment:
275285
msg.append(
276286
f"Environment: {self.current_environment} -> {self.next_environment}"
@@ -293,7 +303,7 @@ def changelog(self):
293303

294304
@property
295305
def comment(self):
296-
return self.changelog
306+
return self.summary
297307

298308
def merge(self, other: Activity) -> ActivityMergeResult:
299309
if not isinstance(other, UpdateActivity):

pkgs/fc/agent/fc/maintenance/maintenance.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -187,7 +187,7 @@ def request_update(log, enc, current_requests) -> Optional[Request]:
187187
"Update preparation was successful. This update will apply "
188188
"changes to the system."
189189
),
190-
_output=activity.changelog,
190+
_output=activity.summary,
191191
current_channel=activity.current_channel_url,
192192
next_channel=activity.next_channel_url,
193193
)

pkgs/fc/agent/fc/util/enc.py

+35-6
Original file line numberDiff line numberDiff line change
@@ -9,7 +9,9 @@
99
from pathlib import Path
1010

1111
import structlog
12+
from fc.util import nixos
1213
from fc.util.directory import connect
14+
from fc.util.time_date import utcnow
1315

1416
structlog = structlog.get_logger()
1517

@@ -112,15 +114,15 @@ def update_enc_nixos_config(log, enc, enc_path):
112114
content=hashlib.sha256(config.encode("utf-8")).hexdigest(),
113115
)
114116
target = os.path.join(basedir, filename)
115-
conditional_update(target, config, encode_json=False)
117+
conditional_update(target, config, mode=0o640, encode_json=False)
116118
os.chown(target, -1, sudo_srv)
117119
previous_files -= {filename}
118120
for filename in previous_files:
119121
log.info("remove-stale-enc-nixos-config", filename=filename)
120122
os.unlink(os.path.join(basedir, filename))
121123

122124

123-
def conditional_update(filename, data, encode_json=True):
125+
def conditional_update(filename, data, mode=0o640, encode_json=True):
124126
"""Updates JSON file on disk only if there is different content."""
125127
with tempfile.NamedTemporaryFile(
126128
mode="w",
@@ -134,7 +136,7 @@ def conditional_update(filename, data, encode_json=True):
134136
else:
135137
tf.write(data)
136138
tf.write("\n")
137-
os.chmod(tf.fileno(), 0o640)
139+
os.chmod(tf.fileno(), mode)
138140
if not (os.path.exists(filename)) or not (filecmp.cmp(filename, tf.name)):
139141
with open(tf.name, "a") as f:
140142
os.fsync(f.fileno())
@@ -158,15 +160,15 @@ def inplace_update(filename, data):
158160
os.fsync(f.fileno())
159161

160162

161-
def retrieve(log, directory_lookup, tgt):
163+
def retrieve(log, func, tgt, mode=0o640):
162164
log.info("retrieve-enc", _replace_msg="Getting: {tgt}", tgt=tgt)
163165
try:
164-
data = directory_lookup()
166+
data = func()
165167
except Exception:
166168
log.error("retrieve-enc-failed", exc_info=True)
167169
return
168170
try:
169-
conditional_update("/etc/nixos/{}".format(tgt), data)
171+
conditional_update("/etc/nixos/{}".format(tgt), data, mode)
170172
except (IOError, OSError):
171173
inplace_update("/etc/nixos/{}".format(tgt), data)
172174

@@ -208,6 +210,32 @@ def load_system_state():
208210
)
209211

210212

213+
def get_release_info(log, enc):
214+
release_info_path = Path("releases.json")
215+
if release_info_path.exists():
216+
with release_info_path.open() as f:
217+
releases = json.load(f)
218+
else:
219+
releases = {}
220+
221+
params = enc["parameters"]
222+
release_name = params.get("release_name")
223+
known_releases = [r["release_name"] for r in releases.values()]
224+
225+
if release_name and release_name not in known_releases:
226+
environment_url = params["environment_url"]
227+
version = nixos.channel_version(environment_url)
228+
releases[version] = {
229+
"environment": params["environment"],
230+
"environment_url": environment_url,
231+
"first_seen_at": utcnow().isoformat(),
232+
"release_name": release_name,
233+
"release_changelog": params.get("release_changelog", ""),
234+
}
235+
236+
return releases
237+
238+
211239
def update_inventory(log, enc):
212240
if (
213241
not enc
@@ -250,6 +278,7 @@ def update_inventory(log, enc):
250278
(lambda: directory.list_service_clients(), "service_clients.json"),
251279
(lambda: directory.list_services(), "services.json"),
252280
(lambda: directory.list_users(), "users.json"),
281+
(lambda: get_release_info(log, enc), "releases.json", 0o644),
253282
],
254283
)
255284

0 commit comments

Comments
 (0)