-
Notifications
You must be signed in to change notification settings - Fork 2.7k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Add automatic check for updates that nags the user when there's a new version available #1429
Changes from 5 commits
e864da9
6e707f3
6c41fee
e2aa1d6
f4c2fe6
6e89da4
0e81f3f
ef68b73
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -106,9 +106,20 @@ yarn_install() { | |
printf "${white}Installing Yarn!$reset\n" | ||
|
||
if [ -d "$HOME/.yarn" ]; then | ||
printf "$red> ~/.yarn already exists, possibly from a past Yarn install.$reset\n" | ||
printf "$red> Remove it (rm -rf ~/.yarn) and run this script again.$reset\n" | ||
exit 0 | ||
if [ -n `which yarn` ]; then | ||
LATEST_VERSION=`curl https://yarnpkg.com/latest-version` | ||
YARN_VERSION=`yarn -V` | ||
|
||
if [ "$LATEST_VERSION" -eq "$YARN_VERSION" ]; then | ||
printf "$green> Yarn is already at the latest version.$reset\n" | ||
else | ||
rm -rf "$HOME/.yarn" | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. 2scary4me, maybe warn the user before doing this. "A newer version of Yarn is available, do you want to delete the existing version and upgrade? (Y/N)" There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Also won't this blow away their config? Or is that stored elsewhere? |
||
fi | ||
else | ||
printf "$red> ~/.yarn already exists, possibly from a past Yarn install.$reset\n" | ||
printf "$red> Remove it (rm -rf ~/.yarn) and run this script again.$reset\n" | ||
exit 0 | ||
fi | ||
fi | ||
|
||
yarn_get_tarball $1 | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -25,8 +25,15 @@ import * as crypto from '../../util/crypto.js'; | |
import map from '../../util/map.js'; | ||
|
||
const invariant = require('invariant'); | ||
const userHome = require('user-home'); | ||
const semver = require('semver'); | ||
const emoji = require('node-emoji'); | ||
const isCI = require('is-ci'); | ||
const path = require('path'); | ||
const fs2 = require('fs'); | ||
|
||
const {verison: YARN_VERSION, installationMethod: YARN_INSTALL_METHOD} = require('../../../package.json'); | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. possibly typo |
||
const ONE_DAY = 1000 * 60 * 60 * 24; | ||
|
||
export type InstallPrepared = { | ||
skip: boolean, | ||
|
@@ -69,6 +76,43 @@ type Flags = { | |
tilde: boolean, | ||
}; | ||
|
||
/** | ||
* Try and detect the installation method for Yarn and provide a command to update it with. | ||
*/ | ||
|
||
function getUpdateCommand(): ?string { | ||
if (YARN_INSTALL_METHOD === 'tar') { | ||
return 'curl -o- -L https://yarnpkg.com/install.sh | bash'; | ||
} | ||
|
||
if (YARN_INSTALL_METHOD === 'homebrwe') { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Typo: "homebrwe" -> "homebrew" |
||
return 'brew upgrade yarn'; | ||
} | ||
|
||
if (YARN_INSTALL_METHOD === 'deb') { | ||
return 'sudo apt-get updat e&& sudo apt-get install yarn'; | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Typo: "updat e" -> "update" |
||
} | ||
|
||
if (YARN_INSTALL_METHOD === 'rpm') { | ||
return 'sudo yum install yarn'; | ||
} | ||
|
||
if (YARN_INSTALL_METHOD === 'npm') { | ||
return 'npm upgrade --global yarn'; | ||
} | ||
|
||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Also add Chocolatey: |
||
return null; | ||
} | ||
|
||
function getUpdateInstaller(): ?string { | ||
// Windows | ||
if (YARN_INSTALL_METHOD === 'msi') { | ||
return 'https://yarnpkg.com/latest.msi'; | ||
} | ||
|
||
return null; | ||
} | ||
|
||
function normalizeFlags(config: Config, rawFlags: Object): Flags { | ||
const flags = { | ||
// install | ||
|
@@ -273,6 +317,8 @@ export class Install { | |
*/ | ||
|
||
async init(): Promise<Array<string>> { | ||
this.checkUpdate(); | ||
|
||
let [depRequests, rawPatterns] = await this.fetchRequestFromCwd(); | ||
const match = await this.matchesIntegrityHash(rawPatterns); | ||
|
||
|
@@ -355,6 +401,7 @@ export class Install { | |
|
||
// fin! | ||
await this.saveLockfileAndIntegrity(rawPatterns); | ||
this.maybeOutputUpdate(); | ||
this.config.requestManager.clearCache(); | ||
return patterns; | ||
} | ||
|
@@ -640,6 +687,72 @@ export class Install { | |
|
||
return request; | ||
} | ||
|
||
/** | ||
* Check for updates every day and output a nag message if there's a newer version. | ||
*/ | ||
|
||
checkUpdate() { | ||
if (!process.stdout.isTTY || isCI) { | ||
// don't show upgrade dialog on CI or non-TTY terminals | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I think this is still valuable for non-TTY environments. Why should we avoid showing a notice about Yarn being outdated just because the user is piping the output to a file, for example? |
||
return; | ||
} | ||
|
||
// only check for updates once a day | ||
const lastUpdateCheck = Number(this.config.getOption('lastUpdateCheck')) || 0; | ||
if (lastUpdateCheck && Date.now() - lastUpdateCheck < ONE_DAY) { | ||
return; | ||
} | ||
|
||
// don't bug for updates on tagged releases | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Hmm, I wonder if we should bug on nightly builds? |
||
if (YARN_VERSION.indexOf('-') >= 0) { | ||
return; | ||
} | ||
|
||
this._checkUpdate().catch(() => { | ||
// swallow errors | ||
}); | ||
} | ||
|
||
async _checkUpdate(): Promise<void> { | ||
let latestVersion = await this.config.requestManager.request({ | ||
url: 'https://yarnpkg.com/latest-version', | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. We need to better automate this getting updated There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Yeah, I'm going to have a webhook that runs when releases are updated, or just a cronjob. The tricky thing is that we can't bump the version number until we verify that all files are attached to the release, and the Windows installer is attached separately from everything else (as it's built on AppVeyor). |
||
}); | ||
invariant(typeof latestVersion === 'string', 'expected string'); | ||
latestVersion = latestVersion.trim(); | ||
if (!semver.valid(latestVersion)) { | ||
return; | ||
} | ||
|
||
// ensure we only check for updates periodically | ||
this.config.registries.yarn.saveHomeConfig({ | ||
lastUpdateCheck: Date.now(), | ||
}); | ||
|
||
if (semver.gt(latestVersion, YARN_VERSION)) { | ||
this.maybeOutputUpdate = () => { | ||
this.reporter.warn(this.reporter.lang('yarnOutdated', latestVersion, YARN_VERSION)); | ||
|
||
const command = getUpdateCommand(); | ||
if (command) { | ||
this.reporter.info(this.reporter.lang('yarnOutdatedCommand')); | ||
this.reporter.command(command); | ||
} else { | ||
const installer = getUpdateInstaller(); | ||
if (installer) { | ||
this.reporter.info(this.reporter.lang('yarnOutdatedInstaller', installer)); | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Optional, but can we use something like boxen (or just render the ANSI codes ourself) to render a fancy box around it, like |
||
} | ||
} | ||
}; | ||
} | ||
} | ||
|
||
/** | ||
* Method to override with a possible upgrade message. | ||
*/ | ||
|
||
maybeOutputUpdate() {} | ||
maybeOutputUpdate: any; | ||
} | ||
|
||
export function setFlags(commander: Object) { | ||
|
This file was deleted.
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -386,6 +386,7 @@ export default class Config { | |
return file; | ||
} | ||
} | ||
|
||
return null; | ||
}); | ||
} | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -77,6 +77,10 @@ const messages = { | |
jsonError: 'Error parsing JSON at $0, $1.', | ||
noFilePermission: "We don't have permissions to touch the file $0.", | ||
|
||
yarnOutdated: "Your current version of Yarn is out of date. The latest version is $0 while you're on $1.", | ||
yarnOutdatedInstaller: 'To upgrade, download the latest installer at $0.', | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. This is totally optional, but we could download the installer and run it for the user. Maybe we'll hold off on that for now and do it as an enhancement! |
||
yarnOutdatedCommand: 'To upgrade, run the following command:', | ||
|
||
tooManyArguments: 'Too many arguments, maximum of $0.', | ||
tooFewArguments: 'Not enough arguments, expected at least $0.', | ||
noArguments: "This command doesn't require any arguments.", | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This may need to handle the
--nightly
flag. We have this inyarn_get_tarball
:You can use https://nightly.yarnpkg.com/latest-tar-version to get the latest nightly version.