diff --git a/lib/updaters/index.js b/lib/updaters/index.js index 47296e3d1..1d3c66ea2 100644 --- a/lib/updaters/index.js +++ b/lib/updaters/index.js @@ -8,6 +8,7 @@ const updatersByType = { csproj: require('./types/csproj'), yaml: require('./types/yaml'), openapi: require('./types/openapi'), + python: require('./types/python'), }; const PLAIN_TEXT_BUMP_FILES = ['VERSION.txt', 'version.txt']; @@ -41,6 +42,9 @@ function getUpdaterByFilename(filename) { if (/\.ya?ml$/.test(filename)) { return getUpdaterByType('yaml'); } + if (/pyproject.toml/.test(filename)) { + return getUpdaterByType('python') + } throw Error( `Unsupported file (${filename}) provided for bumping.\n Please specify the updater \`type\` or use a custom \`updater\`.`, ); diff --git a/lib/updaters/types/python.js b/lib/updaters/types/python.js new file mode 100644 index 000000000..27480c1e9 --- /dev/null +++ b/lib/updaters/types/python.js @@ -0,0 +1,29 @@ +const semverRegex = /version[" ]*=[ ]*["'](.*)["']/i + +const getVersionIndex = function (lines) { + let version + const lineNumber = lines.findIndex(line => { + const found = line.match(semverRegex) + if (found == null) { + return false + } + version = found[1] + return true + }) + return { version, lineNumber } +} + +module.exports.readVersion = function (contents) { + const lines = contents.split('\n') + const versionIndex = getVersionIndex(lines) + return versionIndex.version +} + +module.exports.writeVersion = function (contents, version) { + const lines = contents.split('\n') + const versionIndex = getVersionIndex(lines) + const versionLine = lines[versionIndex.lineNumber] + const newVersionLine = versionLine.replace(versionIndex.version, version) + lines[versionIndex.lineNumber] = newVersionLine + return lines.join('\n') +} \ No newline at end of file diff --git a/test/core.spec.js b/test/core.spec.js index 880220216..6efc268f3 100644 --- a/test/core.spec.js +++ b/test/core.spec.js @@ -1130,6 +1130,43 @@ describe('cli', function () { console.warn = origWarn; } }); + + it('bumps version in Python `pyproject.toml` file', async function () { + const expected = fs.readFileSync( + './test/mocks/pyproject-1.1.0.toml', + 'utf-8', + ); + + const filename = 'python.toml'; + mock({ + bump: 'minor', + realTestFiles: [ + { + filename, + path: './test/mocks/pyproject-1.0.0.toml', + }, + ], + }); + + await exec({ + packageFiles: [{ filename, type: 'python' }], + bumpFiles: [{ filename, type: 'python' }], + }); + + // filePath is the first arg passed to writeFileSync + const packageJsonWriteFileSynchCall = findWriteFileCallForPath({ + writeFileSyncSpy, + filename, + }); + + if (!packageJsonWriteFileSynchCall) { + throw new Error(`writeFileSynch not invoked with path ${filename}`); + } + + const calledWithContentStr = packageJsonWriteFileSynchCall[1]; + expect(calledWithContentStr).toEqual(expected); + }); + }); it('`packageFiles` are bumped along with `bumpFiles` defaults [commit-and-tag-version#533]', async function () { diff --git a/test/mocks/pyproject-1.0.0.toml b/test/mocks/pyproject-1.0.0.toml new file mode 100644 index 000000000..ed8392488 --- /dev/null +++ b/test/mocks/pyproject-1.0.0.toml @@ -0,0 +1,12 @@ +[tool.poetry] +name = "test" +version = "1.0.0" +description = "" +authors = [] + +[tool.poetry.dependencies] +python = "^3.8" + +[build-system] +requires = ["poetry>=1"] +build-backend = "poetry.masonry.api" diff --git a/test/mocks/pyproject-1.1.0.toml b/test/mocks/pyproject-1.1.0.toml new file mode 100644 index 000000000..03745363d --- /dev/null +++ b/test/mocks/pyproject-1.1.0.toml @@ -0,0 +1,12 @@ +[tool.poetry] +name = "test" +version = "1.1.0" +description = "" +authors = [] + +[tool.poetry.dependencies] +python = "^3.8" + +[build-system] +requires = ["poetry>=1"] +build-backend = "poetry.masonry.api"