forked from opensearch-project/OpenSearch-Dashboards
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
[OSCI][FEAT] Changelog Project - PoC Changelog and release notes auto…
…mation tool - OpenSearch Dashboards (opensearch-project#5519) Refactor and Enhance Workflow Management - Added and updated changesets for multiple PRs to improve tracking and documentation of changes. - Removed unnecessary test and dummy files (`test.txt`, various `.yml` fragments) to clean up the repository. - Refactored workflow scripts to streamline changelog generation and fragment handling, moving temporary files to a designated folder. - Updated GitHub Actions workflows by changing event triggers from `pull_request` to `pull_request_target` and vice versa to optimize workflow execution. - Enhanced security and automation by updating token names and adding write permissions to the changeset workflow. - Deleted obsolete workflow file for creating changeset files, now handled by an automated process. - Major clean-up of dummy fragment files and unnecessary changelog entries to maintain clarity and relevancy in documentation. - Implemented minor updates and improvements in codebase, specifically in generating release notes and handling fragments. --------- Signed-off-by: Johnathon Bowers <[email protected]> Signed-off-by: CMDWillYang <[email protected]> Signed-off-by: Qiwen Li <[email protected]> Signed-off-by: qiwen li <[email protected]> Signed-off-by: Samuel Valdes Gutierrez <[email protected]> Signed-off-by: Ashwin P Chandran <[email protected]> Co-authored-by: github-actions[bot] <41898282+github-actions[bot]@users.noreply.github.com> Co-authored-by: Johnathon Bowers <[email protected]> Co-authored-by: CMDWillYang <[email protected]> Co-authored-by: Qiwen Li <[email protected]> Co-authored-by: Ashwin P Chandran <[email protected]> Co-authored-by: Anan Zhuang <[email protected]> Co-authored-by: Josh Romero <[email protected]> Co-authored-by: autochangeset[bot] <154024398+autochangeset[bot]@users.noreply.github.com> Co-authored-by: opensearch-bot[bot] <154024398+opensearch-bot[bot]@users.noreply.github.com> Co-authored-by: opensearch-bot-dev[bot] <154634848+opensearch-bot-dev[bot]@users.noreply.github.com> Co-authored-by: Ashwin P Chandran <[email protected]> Co-authored-by: Miki <[email protected]> Co-authored-by: Kawika Avilla <[email protected]>
- Loading branch information
1 parent
3fa2501
commit 603502b
Showing
7 changed files
with
243 additions
and
1 deletion.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,23 @@ | ||
name: OpenSearch Changelog Workflow | ||
|
||
on: | ||
pull_request_target: | ||
types: [opened, reopened, edited] | ||
|
||
permissions: | ||
contents: read | ||
issues: write | ||
pull-requests: write | ||
|
||
jobs: | ||
update-changelog: | ||
runs-on: ubuntu-latest | ||
steps: | ||
- name: Check out repository | ||
uses: actions/checkout@v4 | ||
- name: Parse changelog entries and submit request for changset creation | ||
uses: BigSamu/[email protected] | ||
with: | ||
token: ${{secrets.GITHUB_TOKEN}} | ||
CHANGELOG_PR_BRIDGE_URL_DOMAIN: ${{secrets.CHANGELOG_PR_BRIDGE_URL_DOMAIN}} | ||
CHANGELOG_PR_BRIDGE_API_KEY: ${{secrets.CHANGELOG_PR_BRIDGE_API_KEY}} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,5 @@ | ||
# Changelog and Release Notes | ||
|
||
For information regarding the changelog and release notes process, please consult the README in the GitHub Actions repository that this process utilizes. To view this README, follow the link below: | ||
|
||
[GitHub Actions Workflow README](https://github.com/BigSamu/OpenSearch_Change_Set_Create_Action/blob/main/README.md) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,8 @@ | ||
/* | ||
* Copyright OpenSearch Contributors | ||
* SPDX-License-Identifier: Apache-2.0 | ||
*/ | ||
|
||
require('../src/setup_node_env'); | ||
require('../src/dev/generate_release_note'); | ||
require('../src/dev/generate_release_note_helper'); |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,134 @@ | ||
/* | ||
* Copyright OpenSearch Contributors | ||
* SPDX-License-Identifier: Apache-2.0 | ||
*/ | ||
|
||
import { join, resolve } from 'path'; | ||
import { readFileSync, writeFileSync, Dirent, rm, rename, promises as fsPromises } from 'fs'; | ||
import { load as loadYaml } from 'js-yaml'; | ||
import { readdir } from 'fs/promises'; | ||
import { version as pkgVersion } from '../../package.json'; | ||
import { | ||
validateFragment, | ||
getCurrentDateFormatted, | ||
Changelog, | ||
SECTION_MAPPING, | ||
fragmentDirPath, | ||
SectionKey, | ||
releaseNotesDirPath, | ||
filePath, | ||
} from './generate_release_note_helper'; | ||
|
||
// Function to add content after the 'Unreleased' section in the changelog | ||
function addContentAfterUnreleased(path: string, newContent: string): void { | ||
let fileContent = readFileSync(path, 'utf8'); | ||
const targetString = '## [Unreleased]'; | ||
const targetIndex = fileContent.indexOf(targetString); | ||
|
||
if (targetIndex !== -1) { | ||
const endOfLineIndex = fileContent.indexOf('\n', targetIndex); | ||
if (endOfLineIndex !== -1) { | ||
fileContent = | ||
fileContent.slice(0, endOfLineIndex + 1) + | ||
'\n' + | ||
newContent + | ||
'\n' + | ||
fileContent.slice(endOfLineIndex + 1); | ||
} else { | ||
throw new Error('End of line for "Unreleased" section not found.'); | ||
} | ||
} else { | ||
throw new Error("'## [Unreleased]' not found in the file."); | ||
} | ||
|
||
writeFileSync(path, fileContent); | ||
} | ||
|
||
async function deleteFragments(fragmentTempDirPath: string) { | ||
rm(fragmentTempDirPath, { recursive: true }, (err: any) => { | ||
if (err) { | ||
throw err; | ||
} | ||
}); | ||
} | ||
|
||
// Read fragment files and populate sections | ||
async function readFragments() { | ||
// Initialize sections | ||
const sections: Changelog = (Object.fromEntries( | ||
Object.keys(SECTION_MAPPING).map((key) => [key, []]) | ||
) as unknown) as Changelog; | ||
|
||
const fragmentPaths = await readdir(fragmentDirPath, { withFileTypes: true }); | ||
for (const fragmentFilename of fragmentPaths) { | ||
// skip non yml or yaml files | ||
if (!/\.ya?ml$/i.test(fragmentFilename.name)) { | ||
// eslint-disable-next-line no-console | ||
console.warn(`Skipping non yml or yaml file ${fragmentFilename.name}`); | ||
continue; | ||
} | ||
|
||
const fragmentPath = join(fragmentDirPath, fragmentFilename.name); | ||
const fragmentContents = readFileSync(fragmentPath, { encoding: 'utf-8' }); | ||
|
||
validateFragment(fragmentContents); | ||
|
||
const fragmentYaml = loadYaml(fragmentContents) as Changelog; | ||
|
||
for (const [sectionKey, entries] of Object.entries(fragmentYaml)) { | ||
sections[sectionKey as SectionKey].push(...entries); | ||
} | ||
} | ||
return { sections, fragmentPaths }; | ||
} | ||
|
||
async function moveFragments(fragmentPaths: Dirent[], fragmentTempDirPath: string): Promise<void> { | ||
// Move fragment files to temp fragments folder | ||
for (const fragmentFilename of fragmentPaths) { | ||
const fragmentPath = resolve(fragmentDirPath, fragmentFilename.name); | ||
const fragmentTempPath = resolve(fragmentTempDirPath, fragmentFilename.name); | ||
rename(fragmentPath, fragmentTempPath, () => {}); | ||
} | ||
} | ||
|
||
function generateChangelog(sections: Changelog) { | ||
// Generate changelog sections | ||
const changelogSections = Object.entries(sections).map(([sectionKey, entries]) => { | ||
const sectionName = SECTION_MAPPING[sectionKey as SectionKey]; | ||
return entries.length === 0 | ||
? `### ${sectionName}` | ||
: `### ${sectionName}\n\n${entries.map((entry) => ` - ${entry}`).join('\n')}`; | ||
}); | ||
|
||
// Generate full changelog | ||
const currentDate = getCurrentDateFormatted(); | ||
const changelog = `## [${pkgVersion}-${currentDate}](https://github.com/opensearch-project/OpenSearch-Dashboards/releases/tag/${pkgVersion})\n\n${changelogSections.join( | ||
'\n\n' | ||
)}`; | ||
// Update changelog file | ||
addContentAfterUnreleased(filePath, changelog); | ||
return changelogSections; | ||
} | ||
|
||
function generateReleaseNote(changelogSections: string[]) { | ||
// Generate release note | ||
const releaseNoteFilename = `opensearch-dashboards.release-notes-${pkgVersion}.md`; | ||
const releaseNoteHeader = `# VERSION ${pkgVersion} Release Note`; | ||
const releaseNote = `${releaseNoteHeader}\n\n${changelogSections.join('\n\n')}`; | ||
writeFileSync(resolve(releaseNotesDirPath, releaseNoteFilename), releaseNote); | ||
} | ||
|
||
(async () => { | ||
const { sections, fragmentPaths } = await readFragments(); | ||
// create folder for temp fragments | ||
const fragmentTempDirPath = await fsPromises.mkdtemp(join(fragmentDirPath, 'tmp_fragments-')); | ||
// move fragments to temp fragments folder | ||
await moveFragments(fragmentPaths, fragmentTempDirPath); | ||
|
||
const changelogSections = generateChangelog(sections); | ||
|
||
generateReleaseNote(changelogSections); | ||
|
||
// remove temp fragments folder | ||
await deleteFragments(fragmentTempDirPath); | ||
})(); |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,59 @@ | ||
/* | ||
* Copyright OpenSearch Contributors | ||
* SPDX-License-Identifier: Apache-2.0 | ||
*/ | ||
|
||
import { resolve } from 'path'; | ||
|
||
export const filePath = resolve(__dirname, '..', '..', 'CHANGELOG.md'); | ||
export const fragmentDirPath = resolve(__dirname, '..', '..', 'changelogs', 'fragments'); | ||
export const releaseNotesDirPath = resolve(__dirname, '..', '..', 'release-notes'); | ||
|
||
export function getCurrentDateFormatted(): string { | ||
return new Date().toISOString().slice(0, 10); | ||
} | ||
|
||
export const SECTION_MAPPING = { | ||
breaking: '💥 Breaking Changes', | ||
deprecate: 'Deprecations', | ||
security: '🛡 Security', | ||
feat: '📈 Features/Enhancements', | ||
fix: '🐛 Bug Fixes', | ||
infra: '🚞 Infrastructure', | ||
doc: '📝 Documentation', | ||
chore: '🛠 Maintenance', | ||
refactor: '🪛 Refactoring', | ||
test: '🔩 Tests', | ||
}; | ||
|
||
export type SectionKey = keyof typeof SECTION_MAPPING; | ||
export type Changelog = Record<SectionKey, string[]>; | ||
|
||
const MAX_ENTRY_LENGTH = 100; | ||
// Each entry must start with '-' and a space, followed by a non-empty string, and be no longer that MAX_ENTRY_LENGTH characters | ||
const entryRegex = new RegExp(`^-.{1,${MAX_ENTRY_LENGTH}}\\(\\[#.+]\\(.+\\)\\)$`); | ||
|
||
// validate format of fragment files | ||
export function validateFragment(content: string) { | ||
const sections = content.split(/(?:\r?\n){2,}/); | ||
|
||
// validate each section | ||
for (const section of sections) { | ||
const lines = section.split('\n'); | ||
const sectionName = lines[0]; | ||
const sectionKey = sectionName.slice(0, -1); | ||
|
||
if (!SECTION_MAPPING[sectionKey as SectionKey] || !sectionName.endsWith(':')) { | ||
throw new Error(`Unknown section ${sectionKey}.`); | ||
} | ||
for (const entry of lines.slice(1)) { | ||
if (entry === '') { | ||
continue; | ||
} | ||
// if (!entryRegex.test(entry)) { | ||
if (!entryRegex.test(entry.trim())) { | ||
throw new Error(`Invalid entry ${entry} in section ${sectionKey}.`); | ||
} | ||
} | ||
} | ||
} |