-
Notifications
You must be signed in to change notification settings - Fork 924
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge branch '2.x' into backport/backport-6395-to-2.x
- Loading branch information
Showing
11 changed files
with
443 additions
and
19 deletions.
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 was deleted.
Oops, something went wrong.
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
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,2 @@ | ||
fix: | ||
- [Discover][Bug] Migrate global state from legacy URL ([#6780](https://github.com/opensearch-project/OpenSearch-Dashboards/pull/6780)) |
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}.`); | ||
} | ||
} | ||
} | ||
} |
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
Oops, something went wrong.