Skip to content

Commit 7f72944

Browse files
committedJan 17, 2025
fix(publish): accept publishConfig.tag to override highes semver check
Setting `tag` in `publishConfig` constitutes a "non default" scenario.
1 parent 7e7961d commit 7f72944

File tree

3 files changed

+45
-34
lines changed

3 files changed

+45
-34
lines changed
 

‎lib/commands/publish.js

+3-3
Original file line numberDiff line numberDiff line change
@@ -117,7 +117,7 @@ class Publish extends BaseCommand {
117117
manifest = await this.#getManifest(spec, opts, true)
118118

119119
const isPreRelease = Boolean(semver.parse(manifest.version).prerelease.length)
120-
const isDefaultTag = this.npm.config.isDefault('tag')
120+
const isDefaultTag = this.npm.config.isDefault('tag') && !manifest.publishConfig?.tag
121121

122122
if (isPreRelease && isDefaultTag) {
123123
throw new Error('You must specify a tag using --tag when publishing a prerelease version.')
@@ -157,7 +157,7 @@ class Publish extends BaseCommand {
157157
}
158158
}
159159

160-
const latestVersion = await this.#latestPublishedVersion(resolved, registry)
160+
const latestVersion = await this.#highestPublishedVersion(resolved, registry)
161161
const latestSemverIsGreater = !!latestVersion && semver.gte(latestVersion, manifest.version)
162162

163163
if (latestSemverIsGreater && isDefaultTag) {
@@ -204,7 +204,7 @@ class Publish extends BaseCommand {
204204
}
205205
}
206206

207-
async #latestPublishedVersion (spec, registry) {
207+
async #highestPublishedVersion (spec, registry) {
208208
try {
209209
const packument = await pacote.packument(spec, {
210210
...this.npm.flatOptions,

‎tap-snapshots/test/lib/commands/publish.js.test.cjs

+14-14
Original file line numberDiff line numberDiff line change
@@ -6,7 +6,7 @@
66
*/
77
'use strict'
88
exports[`test/lib/commands/publish.js TAP _auth config default registry > new package version 1`] = `
9-
+ test-package@1.0.0
9+
+ @npmcli/test-package@1.0.0
1010
`
1111

1212
exports[`test/lib/commands/publish.js TAP bare _auth and registry config > new package version 1`] = `
@@ -15,15 +15,15 @@ exports[`test/lib/commands/publish.js TAP bare _auth and registry config > new p
1515

1616
exports[`test/lib/commands/publish.js TAP dry-run > must match snapshot 1`] = `
1717
Array [
18-
"package: test-package@1.0.0",
18+
"package: @npmcli/test-package@1.0.0",
1919
"Tarball Contents",
20-
"87B package.json",
20+
"95B package.json",
2121
"Tarball Details",
22-
"name: test-package",
22+
"name: @npmcli/test-package",
2323
"version: 1.0.0",
24-
"filename: test-package-1.0.0.tgz",
24+
"filename: npmcli-test-package-1.0.0.tgz",
2525
"package size: {size}",
26-
"unpacked size: 87 B",
26+
"unpacked size: 95 B",
2727
"shasum: {sha}",
2828
"integrity: {integrity}
2929
"total files: 1",
@@ -76,7 +76,7 @@ exports[`test/lib/commands/publish.js TAP has token auth for scope configured re
7676
`
7777

7878
exports[`test/lib/commands/publish.js TAP ignore-scripts > new package version 1`] = `
79-
+ test-package@1.0.0
79+
+ @npmcli/test-package@1.0.0
8080
`
8181

8282
exports[`test/lib/commands/publish.js TAP json > must match snapshot 1`] = `
@@ -87,14 +87,14 @@ Array [
8787

8888
exports[`test/lib/commands/publish.js TAP json > new package json 1`] = `
8989
{
90-
"id": "test-package@1.0.0",
91-
"name": "test-package",
90+
"id": "@npmcli/test-package@1.0.0",
91+
"name": "@npmcli/test-package",
9292
"version": "1.0.0",
9393
"size": "{size}",
94-
"unpackedSize": 87,
94+
"unpackedSize": 95,
9595
"shasum": "{sha}",
9696
"integrity": "{integrity}",
97-
"filename": "test-package-1.0.0.tgz",
97+
"filename": "npmcli-test-package-1.0.0.tgz",
9898
"files": [
9999
{
100100
"path": "package.json",
@@ -249,7 +249,7 @@ Object {
249249
`
250250

251251
exports[`test/lib/commands/publish.js TAP no auth dry-run > must match snapshot 1`] = `
252-
+ test-package@1.0.0
252+
+ @npmcli/test-package@1.0.0
253253
`
254254

255255
exports[`test/lib/commands/publish.js TAP no auth dry-run > warns about auth being needed 1`] = `
@@ -259,7 +259,7 @@ Array [
259259
`
260260

261261
exports[`test/lib/commands/publish.js TAP prioritize CLI flags over publishConfig > new package version 1`] = `
262-
+ test-package@1.0.0
262+
+ @npmcli/test-package@1.0.0
263263
`
264264

265265
exports[`test/lib/commands/publish.js TAP public access > must match snapshot 1`] = `
@@ -285,7 +285,7 @@ exports[`test/lib/commands/publish.js TAP public access > new package version 1`
285285
`
286286

287287
exports[`test/lib/commands/publish.js TAP re-loads publishConfig.registry if added during script process > new package version 1`] = `
288-
+ test-package@1.0.0
288+
+ @npmcli/test-package@1.0.0
289289
`
290290

291291
exports[`test/lib/commands/publish.js TAP respects publishConfig.registry, runs appropriate scripts > new package version 1`] = `

‎test/lib/commands/publish.js

+28-17
Original file line numberDiff line numberDiff line change
@@ -6,7 +6,7 @@ const Arborist = require('@npmcli/arborist')
66
const path = require('node:path')
77
const fs = require('node:fs')
88

9-
const pkg = 'test-package'
9+
const pkg = '@npmcli/test-package'
1010
const token = 'test-auth-token'
1111
const auth = { '//registry.npmjs.org/:_authToken': token }
1212
const alternateRegistry = 'https://other.registry.npmjs.org'
@@ -238,8 +238,7 @@ t.test('throws when invalid tag when not url encodable', async t => {
238238
await t.rejects(
239239
npm.exec('publish', []),
240240
{
241-
/* eslint-disable-next-line max-len */
242-
message: 'Invalid tag name "@test" of package "test-package@@test": Tags may not have any characters that encodeURIComponent encodes.',
241+
message: `Invalid tag name "@test" of package "${pkg}@@test": Tags may not have any characters that encodeURIComponent encodes.`,
243242
}
244243
)
245244
})
@@ -857,15 +856,16 @@ t.test('prerelease dist tag', (t) => {
857856
t.end()
858857
})
859858

860-
t.test('latest dist tag', (t) => {
861-
const init = (version) => ({
859+
t.test('semver highest dist tag', async t => {
860+
const init = ({ version, pkgExtra = {} }) => ({
862861
config: {
863862
loglevel: 'silent',
864863
...auth,
865864
},
866865
prefixDir: {
867866
'package.json': JSON.stringify({
868867
...pkgJson,
868+
...pkgExtra,
869869
version,
870870
}, null, 2),
871871
},
@@ -879,46 +879,57 @@ t.test('latest dist tag', (t) => {
879879
{ version: '105.0.0-pre' },
880880
]
881881

882-
t.test('PREVENTS publish when latest version is HIGHER than publishing version', async t => {
882+
await t.test('PREVENTS publish when highest version is HIGHER than publishing version', async t => {
883883
const version = '99.0.0'
884-
const { npm, registry } = await loadNpmWithRegistry(t, init(version))
884+
const { npm, registry } = await loadNpmWithRegistry(t, init({ version }))
885885
registry.publish(pkg, { noPut: true, packuments })
886886
await t.rejects(async () => {
887887
await npm.exec('publish', [])
888-
/* eslint-disable-next-line max-len */
889888
}, new Error('Cannot implicitly apply the "latest" tag because published version 100.0.0 is higher than the new version 99.0.0. You must specify a tag using --tag.'))
890889
})
891890

892-
t.test('ALLOWS publish when latest is HIGHER than publishing version and flag', async t => {
891+
await t.test('ALLOWS publish when highest is HIGHER than publishing version and flag', async t => {
893892
const version = '99.0.0'
894893
const { npm, registry } = await loadNpmWithRegistry(t, {
895-
...init(version),
894+
...init({ version }),
896895
argv: ['--tag', 'latest'],
897896
})
898897
registry.publish(pkg, { packuments })
899898
await npm.exec('publish', [])
900899
})
901900

902-
t.test('ALLOWS publish when latest versions are LOWER than publishing version', async t => {
901+
await t.test('ALLOWS publish when highest versions are LOWER than publishing version', async t => {
903902
const version = '101.0.0'
904-
const { npm, registry } = await loadNpmWithRegistry(t, init(version))
903+
const { npm, registry } = await loadNpmWithRegistry(t, init({ version }))
905904
registry.publish(pkg, { packuments })
906905
await npm.exec('publish', [])
907906
})
908907

909-
t.test('ALLOWS publish when packument has empty versions (for coverage)', async t => {
908+
await t.test('ALLOWS publish when packument has empty versions (for coverage)', async t => {
910909
const version = '1.0.0'
911-
const { npm, registry } = await loadNpmWithRegistry(t, init(version))
910+
const { npm, registry } = await loadNpmWithRegistry(t, init({ version }))
912911
registry.publish(pkg, { manifest: { versions: { } } })
913912
await npm.exec('publish', [])
914913
})
915914

916-
t.test('ALLOWS publish when packument has empty manifest (for coverage)', async t => {
915+
await t.test('ALLOWS publish when packument has empty manifest (for coverage)', async t => {
917916
const version = '1.0.0'
918-
const { npm, registry } = await loadNpmWithRegistry(t, init(version))
917+
const { npm, registry } = await loadNpmWithRegistry(t, init({ version }))
919918
registry.publish(pkg, { manifest: {} })
920919
await npm.exec('publish', [])
921920
})
922921

923-
t.end()
922+
await t.test('ALLOWS publish when highest version is HIGHER than publishing version with publishConfig', async t => {
923+
const version = '99.0.0'
924+
const { npm, registry } = await loadNpmWithRegistry(t, init({
925+
version,
926+
pkgExtra: {
927+
publishConfig: {
928+
tag: 'next',
929+
},
930+
},
931+
}))
932+
registry.publish(pkg, { packuments })
933+
await npm.exec('publish', [])
934+
})
924935
})

0 commit comments

Comments
 (0)
Please sign in to comment.