Skip to content

Commit 7c79dde

Browse files
committed
#33: Unexpected size for index file [always upload all Packages and Release files]
1 parent 3fedffe commit 7c79dde

9 files changed

+645
-617
lines changed

RaspberryPiDotnetRepository/Data/PackageIndexFile.cs

+1-1
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,7 @@ namespace RaspberryPiDotnetRepository.Data;
33
/// <summary>
44
/// This represents <c>Packages.gz</c> or <c>Packages</c> files in a Debian package repository
55
/// </summary>
6-
public record PackageIndexFile(DebianRelease debianVersion, CpuArchitecture architecture, bool isCompressed, bool isUpToDateInBlobStorage = false) {
6+
public record PackageIndexFile(DebianRelease debianVersion, CpuArchitecture architecture, bool isCompressed) {
77

88
public string filePathRelativeToSuite => Path.Combine("main", $"binary-{architecture.toDebian()}", $"Packages{(isCompressed ? ".gz" : "")}");
99
public string filePathRelativeToRepo => Path.Combine("dists", debianVersion.getCodename(), filePathRelativeToSuite);

RaspberryPiDotnetRepository/Data/ReleaseIndexFile.cs

+1-1
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,7 @@ namespace RaspberryPiDotnetRepository.Data;
33
/// <summary>
44
/// This represents <c>InRelease</c>, <c>Release</c>, or <c>Release.gpg</c> files in a Debian package repository
55
/// </summary>
6-
public record ReleaseIndexFile(DebianRelease debianVersion, bool isUpToDateInBlobStorage = false) {
6+
public class ReleaseIndexFile(DebianRelease debianVersion) {
77

88
private readonly string parentDirectory = Path.Combine("dists", debianVersion.getCodename());
99

Original file line numberDiff line numberDiff line change
@@ -1,3 +1,3 @@
11
namespace RaspberryPiDotnetRepository.Data;
22

3-
public record UploadableFile(string filePathRelativeToRepo, bool isUpToDateInBlobStorage = false);
3+
public record UploadableFile(string filePathRelativeToRepo);

RaspberryPiDotnetRepository/Debian/Repository/ExtraFileGenerator.cs

+31
Original file line numberDiff line numberDiff line change
@@ -14,6 +14,8 @@ public interface ExtraFileGenerator {
1414

1515
string copyGpgPublicKey();
1616

17+
Task<string> generateAddRepoScript();
18+
1719
}
1820

1921
public class ExtraFileGeneratorImpl(IOptions<Options> options, StatisticsService statistics, ILogger<ExtraFileGeneratorImpl> logger): ExtraFileGenerator {
@@ -59,8 +61,37 @@ public string copyGpgPublicKey() {
5961
const string FILENAME = "aldaviva.gpg.key";
6062
string gpgPublicKeyDestination = Path.Combine(options.Value.repositoryBaseDir, FILENAME);
6163
File.Copy(options.Value.gpgPublicKeyPath, gpgPublicKeyDestination, true);
64+
statistics.onFileWritten(gpgPublicKeyDestination);
6265
logger.LogDebug("Wrote GPG public key file Debian repository");
6366
return FILENAME;
6467
}
6568

69+
public async Task<string> generateAddRepoScript() {
70+
const string FILENAME = "addrepo.sh";
71+
string filePath = Path.Combine(options.Value.repositoryBaseDir, FILENAME);
72+
const string CONTENTS = """
73+
#!/bin/sh
74+
75+
echo Adding repository PGP key
76+
sudo wget -q https://raspbian.aldaviva.com/aldaviva.gpg.key -O /etc/apt/trusted.gpg.d/aldaviva.gpg
77+
78+
echo Adding repository
79+
echo "deb https://raspbian.aldaviva.com/ $(lsb_release -cs) main" | sudo tee /etc/apt/sources.list.d/aldaviva.list > /dev/null
80+
81+
echo Finding available packages
82+
sudo apt update
83+
84+
echo Ready to install .NET packages, for example:
85+
echo " sudo apt install dotnet-runtime-latest"
86+
echo " sudo apt install aspnetcore-runtime-latest-lts"
87+
echo " sudo apt install dotnet-sdk-8.0"
88+
echo For more information, see https://github.com/Aldaviva/RaspberryPiDotnetRepository
89+
""";
90+
91+
await File.WriteAllTextAsync(filePath, CONTENTS, Encoding.UTF8);
92+
statistics.onFileWritten(filePath);
93+
logger.LogDebug("Wrote client-side repository installation script");
94+
return FILENAME;
95+
}
96+
6697
}

RaspberryPiDotnetRepository/Debian/Repository/Indexer.cs

+12-18
Original file line numberDiff line numberDiff line change
@@ -50,16 +50,13 @@ public async Task<IEnumerable<IGrouping<DebianRelease, PackageIndexFile>>> gener
5050

5151
private async Task<IEnumerable<PackageIndexFile>> generateIndexOfPackagesInDebianReleaseAndArchitecture(DebianRelease debianRelease, CpuArchitecture cpuArchitecture,
5252
IEnumerable<DebianPackage> debPackages, UpstreamReleasesSecondaryInfo upstreamInfo) {
53-
bool areAllPackagesInIndexUpToDateInBlobStorage = true;
54-
string packageFileContents = string.Join("\n\n", debPackages.Select(package => {
55-
areAllPackagesInIndexUpToDateInBlobStorage &= package.isUpToDateInBlobStorage;
56-
return $"""
57-
{package.getControl(upstreamInfo).serialize().Trim()}
58-
Filename: {package.filePathRelativeToRepo}
59-
Size: {package.downloadSize.ConvertToUnit(Unit.Byte).Quantity:F0}
60-
SHA256: {package.fileHashSha256}
61-
""".ReplaceLineEndings("\n");
62-
}));
53+
string packageFileContents = string.Join("\n\n", debPackages.Select(package =>
54+
$"""
55+
{package.getControl(upstreamInfo).serialize().Trim()}
56+
Filename: {package.filePathRelativeToRepo}
57+
Size: {package.downloadSize.ConvertToUnit(Unit.Byte).Quantity:F0}
58+
SHA256: {package.fileHashSha256}
59+
""".ReplaceLineEndings("\n")));
6360

6461
string packageFileRelativeToSuite = Path.Combine("main", $"binary-{cpuArchitecture.toDebian()}", "Packages");
6562
string packageFileAbsolutePath = Path.Combine(options.Value.repositoryBaseDir, "dists", debianRelease.getCodename(), packageFileRelativeToSuite);
@@ -79,22 +76,19 @@ private async Task<IEnumerable<PackageIndexFile>> generateIndexOfPackagesInDebia
7976
logger.LogInformation("Generated Packages index files for Debian {debian} {arch}", debianRelease.getCodename(), cpuArchitecture.toDebian());
8077

8178
return [
82-
new PackageIndexFile(debianRelease, cpuArchitecture, true, areAllPackagesInIndexUpToDateInBlobStorage),
83-
new PackageIndexFile(debianRelease, cpuArchitecture, false, areAllPackagesInIndexUpToDateInBlobStorage)
79+
new PackageIndexFile(debianRelease, cpuArchitecture, true),
80+
new PackageIndexFile(debianRelease, cpuArchitecture, false)
8481
];
8582
}
8683

8784
public async Task<ReleaseIndexFile> generateReleaseIndex(DebianRelease debianRelease, IEnumerable<PackageIndexFile> packageIndexFiles) {
88-
string repositoryBaseDir = options.Value.repositoryBaseDir;
89-
bool areAllReleaseIndexFilesUpToDateInBlobStorage = true;
85+
string repositoryBaseDir = options.Value.repositoryBaseDir;
9086

9187
string indexSha256Hashes = string.Join('\n', await Task.WhenAll(packageIndexFiles.Select(async packageIndexFile => {
9288
await using FileStream fileStream = File.OpenRead(Path.Combine(repositoryBaseDir, packageIndexFile.filePathRelativeToRepo));
9389

9490
string sha256Hash = Convert.ToHexString(await SHA256.HashDataAsync(fileStream)).ToLowerInvariant();
95-
long fileSize = fileStream.Length;
96-
areAllReleaseIndexFilesUpToDateInBlobStorage &= packageIndexFile.isUpToDateInBlobStorage;
97-
return $" {sha256Hash} {fileSize:D} {Paths.Dos2UnixSlashes(packageIndexFile.filePathRelativeToSuite)}";
91+
return $" {sha256Hash} {fileStream.Length:D} {Paths.Dos2UnixSlashes(packageIndexFile.filePathRelativeToSuite)}";
9892
})));
9993

10094
string releaseFileCleartext =
@@ -111,7 +105,7 @@ public async Task<ReleaseIndexFile> generateReleaseIndex(DebianRelease debianRel
111105
{indexSha256Hashes}
112106
""".ReplaceLineEndings("\n");
113107

114-
ReleaseIndexFile releaseIndexFile = new(debianRelease, areAllReleaseIndexFilesUpToDateInBlobStorage);
108+
ReleaseIndexFile releaseIndexFile = new(debianRelease);
115109

116110
string filePath = Path.Combine(repositoryBaseDir, releaseIndexFile.releaseFilePathRelativeToRepo);
117111
await File.WriteAllTextAsync(filePath, releaseFileCleartext, Encoding.UTF8); // Bom.Squad has already defused this UTF-8 BOM, or else apt will get its limbs blown off

RaspberryPiDotnetRepository/Orchestrator.cs

+28-28
Original file line numberDiff line numberDiff line change
@@ -11,18 +11,18 @@
1111
namespace RaspberryPiDotnetRepository;
1212

1313
public class Orchestrator(
14-
SdkDownloader sdkDownloader,
15-
PackageRequester packageRequester,
16-
PackageGenerator packageGenerator,
17-
Indexer indexer,
18-
CdnClient cdnClient,
19-
ExtraFileGenerator extraFileGenerator,
20-
ManifestManager manifestManager,
21-
BlobStorageClient blobStorage,
22-
StatisticsService statistics,
14+
SdkDownloader sdkDownloader,
15+
PackageRequester packageRequester,
16+
PackageGenerator packageGenerator,
17+
Indexer indexer,
18+
CdnClient cdnClient,
19+
ExtraFileGenerator extraFileGenerator,
20+
ManifestManager manifestManager,
21+
BlobStorageClient blobStorage,
22+
StatisticsService statistics,
2323
IHostApplicationLifetime appLifetime,
24-
IOptions<Options> options,
25-
ILogger<Orchestrator> logger
24+
IOptions<Options> options,
25+
ILogger<Orchestrator> logger
2626
): BackgroundService {
2727

2828
private static readonly Version MIN_DOTNET_MINOR_VERSION = new(6, 0);
@@ -55,42 +55,42 @@ protected override async Task ExecuteAsync(CancellationToken ct) {
5555
ReleaseIndexFile[] releaseIndexFiles = await Task.WhenAll(packageIndicesByDebianRelease.Select(releaseFiles => indexer.generateReleaseIndex(releaseFiles.Key, releaseFiles)));
5656

5757
// Write readme, badges, and GPG public key
58-
string readmeFilename = await extraFileGenerator.generateReadme();
59-
IEnumerable<UploadableFile> badgeFiles = await extraFileGenerator.generateReadmeBadges(upstreamReleases.First(release => release.isLatestMinorVersion));
60-
string gpgPublicKeyFile = extraFileGenerator.copyGpgPublicKey();
58+
string readmeFilename = await extraFileGenerator.generateReadme();
59+
IEnumerable<UploadableFile> badgeFiles = await extraFileGenerator.generateReadmeBadges(upstreamReleases.First(release => release.isLatestMinorVersion));
60+
string gpgPublicKeyFile = extraFileGenerator.copyGpgPublicKey();
61+
string addRepoScriptFile = await extraFileGenerator.generateAddRepoScript();
6162

6263
// Upload .deb packages to Azure Blob Storage
6364
string repoBaseDir = options.Value.repositoryBaseDir;
6465
await Task.WhenAll(generatedPackages.Where(p => !p.isUpToDateInBlobStorage).Select(p =>
6566
blobStorage.uploadFile(Path.Combine(repoBaseDir, p.filePathRelativeToRepo), p.filePathRelativeToRepo, "application/vnd.debian.binary-package", ct)));
6667

6768
// Upload Packages.gz indices to Azure Blob Storage
68-
Task<BlobContentInfo?[]> packageIndexUploads = Task.WhenAll(packageIndicesByDebianRelease.SelectMany(debianRelease => debianRelease).Where(file => !file.isUpToDateInBlobStorage).Select(
69-
packageIndexFile =>
70-
blobStorage.uploadFile(Path.Combine(repoBaseDir, packageIndexFile.filePathRelativeToRepo), packageIndexFile.filePathRelativeToRepo,
71-
packageIndexFile.isCompressed ? "application/gzip" : "text/plain", ct)));
69+
Task<BlobContentInfo?[]> packageIndexUploads = Task.WhenAll(packageIndicesByDebianRelease.SelectMany(debianRelease => debianRelease).Select(packageIndexFile =>
70+
blobStorage.uploadFile(Path.Combine(repoBaseDir, packageIndexFile.filePathRelativeToRepo), packageIndexFile.filePathRelativeToRepo,
71+
packageIndexFile.isCompressed ? "application/gzip" : "text/plain", ct)));
7272

73-
// Upload InRelease index files to Azure Blob Storage
74-
Task<BlobContentInfo?[]> releaseIndexUploads = Task.WhenAll(releaseIndexFiles.Where(file => !file.isUpToDateInBlobStorage).SelectMany(file =>
73+
// Upload InRelease indices to Azure Blob Storage
74+
Task<BlobContentInfo?[]> releaseIndexUploads = Task.WhenAll(releaseIndexFiles.SelectMany(file =>
7575
new[] { file.inreleaseFilePathRelativeToRepo, file.releaseFilePathRelativeToRepo, file.releaseGpgFilePathRelativeToRepo }.Select(relativeFilePath =>
76-
blobStorage.uploadFile(Path.Combine(repoBaseDir, relativeFilePath), relativeFilePath, Path.GetExtension(relativeFilePath) == ".gpg" ? "application/pgp-signature" : "text/plain",
77-
ct))));
76+
blobStorage.uploadFile(Path.Combine(repoBaseDir, relativeFilePath), relativeFilePath,
77+
".gpg".Equals(Path.GetExtension(relativeFilePath), StringComparison.OrdinalIgnoreCase) ? "application/pgp-signature" : "text/plain", ct))));
7878

7979
await packageIndexUploads;
8080
await releaseIndexUploads;
8181

82-
// Upload badge JSON files to Azure Blob Storage
83-
await Task.WhenAll(badgeFiles.Where(file => !file.isUpToDateInBlobStorage)
84-
.Select(file => blobStorage.uploadFile(Path.Combine(repoBaseDir, file.filePathRelativeToRepo), file.filePathRelativeToRepo, "application/json", ct)));
82+
// Upload badge JSON and other extra files to Azure Blob Storage
83+
await Task.WhenAll(badgeFiles.Select(file => blobStorage.uploadFile(Path.Combine(repoBaseDir, file.filePathRelativeToRepo), file.filePathRelativeToRepo, "application/json", ct)));
8584
await blobStorage.uploadFile(Path.Combine(repoBaseDir, readmeFilename), readmeFilename, "text/plain", ct);
8685
await blobStorage.uploadFile(Path.Combine(repoBaseDir, gpgPublicKeyFile), gpgPublicKeyFile, "application/pgp-keys", ct);
87-
88-
// Clear CDN cache
89-
await cdnClient.purge(["/dists/*", "/badges/*", "/manifest.json"]);
86+
await blobStorage.uploadFile(Path.Combine(repoBaseDir, addRepoScriptFile), addRepoScriptFile, "application/x-sh", ct);
9087

9188
// Upload manifest.json file to Azure Blob Storage
9289
await blobStorage.uploadFile(manifestManager.manifestFilePath, manifestManager.manifestFilename, "application/json", ct);
9390

91+
// Clear CDN cache
92+
await cdnClient.purge(["/dists/*", "/badges/*", "/manifest.json"]);
93+
9494
// Delete outdated .deb package files from Azure Blob Storage
9595
await Task.WhenAll(oldManifest?.packages.Except(newManifest.packages).Select(packageToDelete => blobStorage.deleteFile(packageToDelete.filePathRelativeToRepo, ct)) ?? []);
9696

RaspberryPiDotnetRepository/Program.cs

+2-1
Original file line numberDiff line numberDiff line change
@@ -17,14 +17,15 @@
1717
using PGP = Unfucked.PGP;
1818

1919
#pragma warning disable CS8634 // The type cannot be used as type parameter in the generic type or method. Nullability of type argument doesn't match 'class' constraint. - You can safely try to register a singleton with a null value, it just won't register. Just make sure you inject it as nullable.
20+
#pragma warning disable IDE0001 // same as above
2021

2122
// apt will get its limbs blown off with "Clearsigned file isn't valid, got 'NOSPLIT' (does the network require authentication?)" if InRelease starts with UTF-8 BOM
2223
BomSquad.DefuseUtf8Bom();
2324

2425
HostApplicationBuilder appConfig = Host.CreateApplicationBuilder(args);
2526
appConfig.Configuration.AlsoSearchForJsonFilesInExecutableDirectory();
2627

27-
appConfig.Logging.AddUnfuckedConsole(options => options.IncludeNamespaces = false);
28+
appConfig.Logging.AddUnfuckedConsole();
2829

2930
appConfig.Services
3031
.Configure<Options>(appConfig.Configuration)

RaspberryPiDotnetRepository/RaspberryPiDotnetRepository.csproj

+3-1
Original file line numberDiff line numberDiff line change
@@ -5,7 +5,7 @@
55
<RuntimeIdentifier>win-x64</RuntimeIdentifier>
66
<ImplicitUsings>enable</ImplicitUsings>
77
<Nullable>enable</Nullable>
8-
<Version>1.1.0</Version>
8+
<Version>1.1.1</Version>
99
<Authors>Ben Hutchison</Authors>
1010
<Copyright>© 2024 $(Authors)</Copyright>
1111
<Company>$(Authors)</Company>
@@ -14,6 +14,8 @@
1414
<RestorePackagesWithLockFile>true</RestorePackagesWithLockFile>
1515
<ApplicationIcon>rpi-dotnet.ico</ApplicationIcon>
1616
<NoWarn>$(NoWarn);IDE0079;8524</NoWarn>
17+
<PublishSingleFile>true</PublishSingleFile>
18+
<SelfContained>false</SelfContained>
1719
</PropertyGroup>
1820

1921
<ItemGroup>

0 commit comments

Comments
 (0)