2024-05-19 00:57:11 +02:00
|
|
|
const https = require('https');
|
|
|
|
const crypto = require('crypto');
|
|
|
|
const fs = require('fs');
|
|
|
|
const { URL } = require('url');
|
|
|
|
|
2024-08-15 20:36:35 +02:00
|
|
|
const repository = process.env.GITHUB_REPOSITORY;
|
|
|
|
const version = process.env.VERSION;
|
2024-08-25 19:12:24 +00:00
|
|
|
const targetAbi = "10.9.10.0";
|
2024-08-15 20:36:35 +02:00
|
|
|
|
2024-05-19 00:57:11 +02:00
|
|
|
// Read manifest.json
|
|
|
|
const manifestPath = './manifest.json';
|
|
|
|
if (!fs.existsSync(manifestPath)) {
|
|
|
|
console.error('manifest.json file not found');
|
|
|
|
process.exit(1);
|
|
|
|
}
|
2024-08-15 23:53:41 +02:00
|
|
|
|
|
|
|
// Read README.md
|
|
|
|
const readmePath = './README.md';
|
|
|
|
if (!fs.existsSync(readmePath)) {
|
|
|
|
console.error('README.md file not found');
|
|
|
|
process.exit(1);
|
|
|
|
}
|
|
|
|
|
2024-05-19 00:57:11 +02:00
|
|
|
const jsonData = JSON.parse(fs.readFileSync(manifestPath, 'utf8'));
|
|
|
|
|
|
|
|
const newVersion = {
|
2024-08-15 20:36:35 +02:00
|
|
|
version,
|
2024-08-15 21:30:47 +02:00
|
|
|
changelog: `- See the full changelog at [GitHub](https://github.com/${repository}/releases/tag/10.9/v${version})\n`,
|
2024-08-15 20:36:35 +02:00
|
|
|
targetAbi,
|
|
|
|
sourceUrl: `https://github.com/${repository}/releases/download/10.9/v${version}/intro-skipper-v${version}.zip`,
|
|
|
|
checksum: getMD5FromFile(),
|
|
|
|
timestamp: new Date().toISOString().replace(/\.\d{3}Z$/, 'Z')
|
2024-05-19 00:57:11 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
async function updateManifest() {
|
2024-05-26 01:11:38 +02:00
|
|
|
await validVersion(newVersion);
|
|
|
|
|
2024-05-19 00:57:11 +02:00
|
|
|
// Add the new version to the manifest
|
|
|
|
jsonData[0].versions.unshift(newVersion);
|
|
|
|
|
|
|
|
console.log('Manifest updated successfully.');
|
2024-08-15 23:53:41 +02:00
|
|
|
updateReadMeVersion();
|
2024-08-23 22:51:42 +02:00
|
|
|
|
|
|
|
cleanUpOldReleases();
|
|
|
|
|
|
|
|
// Write the modified JSON data back to the file
|
|
|
|
fs.writeFileSync(manifestPath, JSON.stringify(jsonData, null, 4), 'utf8');
|
|
|
|
|
2024-08-15 19:41:15 +02:00
|
|
|
process.exit(0); // Exit with no error
|
2024-05-19 00:57:11 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
async function validVersion(version) {
|
|
|
|
console.log(`Validating version ${version.version}...`);
|
|
|
|
|
|
|
|
const isValidChecksum = await verifyChecksum(version.sourceUrl, version.checksum);
|
|
|
|
if (!isValidChecksum) {
|
|
|
|
console.error(`Checksum mismatch for URL: ${version.sourceUrl}`);
|
|
|
|
process.exit(1); // Exit with an error code
|
|
|
|
} else {
|
|
|
|
console.log(`Version ${version.version} is valid.`);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
async function verifyChecksum(url, expectedChecksum) {
|
|
|
|
try {
|
2024-08-15 19:41:15 +02:00
|
|
|
const hash = await downloadAndHashFile(url);
|
2024-05-19 00:57:11 +02:00
|
|
|
return hash === expectedChecksum;
|
|
|
|
} catch (error) {
|
|
|
|
console.error(`Error verifying checksum for URL: ${url}`, error);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-08-15 19:41:15 +02:00
|
|
|
async function downloadAndHashFile(url, redirects = 5) {
|
2024-05-19 00:57:11 +02:00
|
|
|
if (redirects === 0) {
|
|
|
|
throw new Error('Too many redirects');
|
|
|
|
}
|
|
|
|
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
https.get(url, (response) => {
|
|
|
|
if (response.statusCode >= 300 && response.statusCode < 400 && response.headers.location) {
|
|
|
|
// Follow redirect
|
|
|
|
const redirectUrl = new URL(response.headers.location, url).toString();
|
2024-08-15 19:41:15 +02:00
|
|
|
downloadAndHashFile(redirectUrl, redirects - 1)
|
2024-05-19 00:57:11 +02:00
|
|
|
.then(resolve)
|
|
|
|
.catch(reject);
|
|
|
|
} else if (response.statusCode === 200) {
|
2024-08-15 19:41:15 +02:00
|
|
|
const hash = crypto.createHash('md5');
|
|
|
|
response.pipe(hash);
|
|
|
|
response.on('end', () => {
|
|
|
|
resolve(hash.digest('hex'));
|
|
|
|
});
|
|
|
|
response.on('error', (err) => {
|
|
|
|
reject(err);
|
2024-05-19 00:57:11 +02:00
|
|
|
});
|
|
|
|
} else {
|
|
|
|
reject(new Error(`Failed to get '${url}' (${response.statusCode})`));
|
|
|
|
}
|
|
|
|
}).on('error', (err) => {
|
2024-08-15 19:41:15 +02:00
|
|
|
reject(err);
|
2024-05-19 00:57:11 +02:00
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2024-08-15 20:36:35 +02:00
|
|
|
function getMD5FromFile() {
|
|
|
|
const fileBuffer = fs.readFileSync(`intro-skipper-v${version}.zip`);
|
|
|
|
return crypto.createHash('md5').update(fileBuffer).digest('hex');
|
|
|
|
}
|
|
|
|
|
2024-08-15 23:53:41 +02:00
|
|
|
function getReadMeVersion() {
|
|
|
|
let parts = targetAbi.split('.').map(Number);
|
|
|
|
parts.pop();
|
|
|
|
return parts.join(".");
|
|
|
|
}
|
|
|
|
|
|
|
|
function updateReadMeVersion() {
|
|
|
|
const newVersion = getReadMeVersion();
|
|
|
|
const readMeContent = fs.readFileSync(readmePath, 'utf8');
|
|
|
|
|
|
|
|
const updatedContent = readMeContent
|
|
|
|
.replace(/Jellyfin.*\(or newer\)/, `Jellyfin ${newVersion} (or newer)`)
|
|
|
|
if (readMeContent != updatedContent) {
|
|
|
|
fs.writeFileSync(readmePath, updatedContent);
|
|
|
|
console.log('Updated README with new Jellyfin version.');
|
|
|
|
} else {
|
|
|
|
console.log('README has already newest Jellyfin version.');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-08-23 22:51:42 +02:00
|
|
|
function cleanUpOldReleases() {
|
|
|
|
// Extract all unique targetAbi values
|
|
|
|
const abiSet = new Set();
|
|
|
|
jsonData.forEach(entry => {
|
|
|
|
entry.versions.forEach(version => {
|
|
|
|
abiSet.add(version.targetAbi);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
// Convert the Set to an array and sort it in descending order
|
|
|
|
const abiArray = Array.from(abiSet).sort((a, b) => {
|
|
|
|
const aParts = a.split('.').map(Number);
|
|
|
|
const bParts = b.split('.').map(Number);
|
|
|
|
|
|
|
|
for (let i = 0; i < aParts.length; i++) {
|
|
|
|
if (aParts[i] > bParts[i]) return -1;
|
|
|
|
if (aParts[i] < bParts[i]) return 1;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
});
|
|
|
|
|
|
|
|
// Identify the highest and second highest targetAbi
|
|
|
|
const highestAbi = abiArray[0];
|
|
|
|
const secondHighestAbi = abiArray[1];
|
|
|
|
|
|
|
|
// Filter the versions array to keep only those with the highest or second highest targetAbi
|
|
|
|
jsonData.forEach(entry => {
|
|
|
|
entry.versions = entry.versions.filter(version =>
|
|
|
|
version.targetAbi === highestAbi || version.targetAbi === secondHighestAbi
|
|
|
|
);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2024-05-19 00:57:11 +02:00
|
|
|
async function run() {
|
2024-05-26 01:11:38 +02:00
|
|
|
await updateManifest();
|
2024-05-19 00:57:11 +02:00
|
|
|
}
|
|
|
|
|
2024-05-27 18:24:05 +02:00
|
|
|
run();
|