diff --git a/src/client/extensionActivation.ts b/src/client/extensionActivation.ts index 37ca1ad54afc..c4b663fdba6d 100644 --- a/src/client/extensionActivation.ts +++ b/src/client/extensionActivation.ts @@ -51,7 +51,7 @@ import { IInterpreterQuickPick } from './interpreter/configuration/types'; import { registerAllCreateEnvironmentFeatures } from './pythonEnvironments/creation/registrations'; import { registerCreateEnvironmentTriggers } from './pythonEnvironments/creation/createEnvironmentTrigger'; import { initializePersistentStateForTriggers } from './common/persistentState'; -import { logAndNotifyOnFormatterSetting } from './logging/settingLogs'; +import { logAndNotifyOnLegacySettings } from './logging/settingLogs'; export async function activateComponents( // `ext` is passed to any extra activation funcs. @@ -183,7 +183,7 @@ async function activateLegacy(ext: ExtensionState): Promise { ), ); - logAndNotifyOnFormatterSetting(); + logAndNotifyOnLegacySettings(); registerCreateEnvironmentTriggers(disposables); initializePersistentStateForTriggers(ext.context); } diff --git a/src/client/logging/settingLogs.ts b/src/client/logging/settingLogs.ts index 721ab80d4500..da3f8f9d5118 100644 --- a/src/client/logging/settingLogs.ts +++ b/src/client/logging/settingLogs.ts @@ -6,10 +6,9 @@ import { traceError, traceInfo } from '.'; import { Commands, PVSC_EXTENSION_ID } from '../common/constants'; import { showErrorMessage } from '../common/vscodeApis/windowApis'; import { getConfiguration, getWorkspaceFolders } from '../common/vscodeApis/workspaceApis'; -import { Common } from '../common/utils/localize'; -import { executeCommand } from '../common/vscodeApis/commandApis'; -export function logAndNotifyOnFormatterSetting(): void { +function logOnLegacyFormatterSetting(): boolean { + let usesLegacyFormatter = false; getWorkspaceFolders()?.forEach(async (workspace) => { let config = getConfiguration('editor', { uri: workspace.uri, languageId: 'python' }); if (!config) { @@ -21,22 +20,86 @@ export function logAndNotifyOnFormatterSetting(): void { const formatter = config.get('defaultFormatter', ''); traceInfo(`Default formatter is set to ${formatter} for workspace ${workspace.uri.fsPath}`); if (formatter === PVSC_EXTENSION_ID) { + usesLegacyFormatter = true; traceError('Formatting features have been moved to separate formatter extensions.'); + traceError('See here for more information: https://code.visualstudio.com/docs/python/formatting'); traceError('Please install the formatter extension you prefer and set it as the default formatter.'); traceError('For `autopep8` use: https://marketplace.visualstudio.com/items?itemName=ms-python.autopep8'); traceError( 'For `black` use: https://marketplace.visualstudio.com/items?itemName=ms-python.black-formatter', ); traceError('For `yapf` use: https://marketplace.visualstudio.com/items?itemName=eeyore.yapf'); - const response = await showErrorMessage( - l10n.t( - 'Formatting features have been moved to separate formatter extensions. Please install the formatter extension you prefer and set it as the default formatter.', - ), - Common.showLogs, - ); - if (response === Common.showLogs) { - executeCommand(Commands.ViewOutput); + } + }); + return usesLegacyFormatter; +} + +function logOnLegacyLinterSetting(): boolean { + let usesLegacyLinter = false; + getWorkspaceFolders()?.forEach(async (workspace) => { + let config = getConfiguration('python', { uri: workspace.uri, languageId: 'python' }); + if (!config) { + config = getConfiguration('python', workspace.uri); + if (!config) { + traceError('Unable to get editor configuration'); } } + + const linters: string[] = [ + 'pylint', + 'flake8', + 'mypy', + 'pydocstyle', + 'pylama', + 'pycodestyle', + 'bandit', + 'prospector', + ]; + + linters.forEach((linter) => { + const linterEnabled = config.get(`linting.${linter}Enabled`, false); + if (linterEnabled) { + usesLegacyLinter = true; + traceError('Linting features have been moved to separate linter extensions.'); + traceError('See here for more information: https://code.visualstudio.com/docs/python/linting'); + if (linter === 'pylint' || linter === 'flake8') { + traceError( + `Please install "${linter}" extension: https://marketplace.visualstudio.com/items?itemName=ms-python.${linter}`, + ); + } else if (linter === 'mypy') { + traceError( + `Please install "${linter}" extension: https://marketplace.visualstudio.com/items?itemName=ms-python.mypy-type-checker`, + ); + } else if (['pydocstyle', 'pylama', 'pycodestyle', 'bandit'].includes(linter)) { + traceError( + `selected linter "${linter}" may be supported by extensions like "Ruff", which include several linter rules: https://marketplace.visualstudio.com/items?itemName=charliermarsh.ruff`, + ); + } + } + }); }); + + return usesLegacyLinter; +} + +let _isShown = false; +async function notifyLegacySettings(): Promise { + if (_isShown) { + return; + } + _isShown = true; + showErrorMessage( + l10n.t( + `Formatting and linting features have been deprecated from the Python extension. Please install a linter or a formatter extension. [Open logs](command:${Commands.ViewOutput}) for more information.`, + ), + ); +} + +export function logAndNotifyOnLegacySettings(): void { + const usesLegacyFormatter = logOnLegacyFormatterSetting(); + const usesLegacyLinter = logOnLegacyLinterSetting(); + + if (usesLegacyFormatter || usesLegacyLinter) { + setImmediate(() => notifyLegacySettings().ignoreErrors()); + } }