Skip to content
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

Use Python Extension's getExecutionDetails to provide venv to reader #201

Merged
merged 46 commits into from
Mar 30, 2021
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
46 commits
Select commit Hold shift + click to select a range
f8e62ba
Get reader Python binary with Python Extension API
rogermparent Mar 23, 2021
ae75cb8
Properly fall back to `dvc` with no Python extension venv
rogermparent Mar 23, 2021
341140c
Remove python extension call in extension tests
rogermparent Mar 23, 2021
23c5e76
Merge branch 'master' of github.com:iterative/vscode-dvc into python-…
rogermparent Mar 23, 2021
38dd8f2
Revert jest -> mocha exp test move
rogermparent Mar 23, 2021
96f4d63
Support cliPath option to override Python env
rogermparent Mar 23, 2021
4502e55
Use optional chaining instead of early returns
rogermparent Mar 23, 2021
788b0d0
Make getExecutionDetails a one-liner and re-add early return for getR…
rogermparent Mar 23, 2021
df67cf5
Merge branch 'master' of github.com:iterative/vscode-dvc into python-…
rogermparent Mar 23, 2021
91b6ff8
Replace redundant createPythonInstance with createInstance call
rogermparent Mar 23, 2021
286bdd1
Make cliPath optional in most interfaces
rogermparent Mar 24, 2021
cd324ab
Make setting default dvc path set undefined
rogermparent Mar 24, 2021
e04e954
Add extra implementation to vscode mock
rogermparent Mar 24, 2021
78d1f98
Change dvcPath config option description
rogermparent Mar 24, 2021
f388299
Remove `findCliPath` and Rework `selectDvcPath`
rogermparent Mar 25, 2021
4bfbb56
Simplify custom dvc binary callback
rogermparent Mar 25, 2021
a4a6636
Remove object dvcclipath and use config api
rogermparent Mar 25, 2021
a775b12
Set dvcPathStatusBarItem text in status bar constructor
rogermparent Mar 25, 2021
d675761
Merge branch 'master' into python-extension-reader-venv
rogermparent Mar 25, 2021
3c1c278
Merge branch 'spike-decoration-provider' of github.com:iterative/vsco…
rogermparent Mar 25, 2021
73f3305
Merge branch 'master' of github.com:iterative/vscode-dvc into python-…
rogermparent Mar 25, 2021
1597231
Rewrite dvcPath description and move it to nls file
rogermparent Mar 25, 2021
f3d2d9d
Move pythonExtension
rogermparent Mar 25, 2021
a628a13
Add tests for `getDvcInvocation`
rogermparent Mar 25, 2021
f4e3769
Change DVC Path to DVC CLI Path in config
rogermparent Mar 25, 2021
6dc58a4
Roll getActivatedPythonExtension into getReadyPythonExtension
rogermparent Mar 25, 2021
a57fa12
Remove "Global" option in DVC path selector
rogermparent Mar 25, 2021
f446e53
Remove vscode mock additions
rogermparent Mar 25, 2021
df0a3ac
Remove DVCPATH and dev-ui-win
rogermparent Mar 26, 2021
f5fb440
Change DVC path selector text entry to file picker
rogermparent Mar 26, 2021
4994041
Remove custom dvc item integration test
rogermparent Mar 26, 2021
aedb01a
Fix typo
rogermparent Mar 26, 2021
a3a69b7
Merge branch 'master' of github.com:iterative/vscode-dvc into python-…
rogermparent Mar 26, 2021
bd34268
Remove unused return value on setDvcPath and remove broken test tempo…
rogermparent Mar 29, 2021
ce701e7
Re-add new broken vscode test
rogermparent Mar 29, 2021
239e5c4
import with `* as` and combine `resolves` into one line.
rogermparent Mar 29, 2021
551a18c
fix tests
mattseddon Mar 29, 2021
03fb001
Merge branch 'master' into python-extension-reader-venv
rogermparent Mar 29, 2021
b110507
Replace delay in extension test with awaited event
rogermparent Mar 30, 2021
06d4c80
Refactor out config change promise into reusable function
rogermparent Mar 30, 2021
ce693c7
Improve configChangePromise
rogermparent Mar 30, 2021
da193c4
Slightly reorder calls and run Prettier
rogermparent Mar 30, 2021
da11c1e
binary -> executable
rogermparent Mar 30, 2021
de5320c
Use blank string fallback on dvcPath
rogermparent Mar 30, 2021
cd689bd
Re-add dvcPath arg to findDvcRootPaths
rogermparent Mar 30, 2021
42d4e87
Make CLI path non-optional
rogermparent Mar 30, 2021
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
4 changes: 0 additions & 4 deletions .vscode/launch.json
Original file line number Diff line number Diff line change
Expand Up @@ -12,15 +12,13 @@
"--disable-extensions"
],
"env": {
"DVCPATH": ".env/bin/dvc",
"HOT_RELOAD": "true",
"PYTHONPATH": ".env/bin/python3",
"USE_DEV_UI": "true"
},
"outFiles": ["${workspaceFolder}/extension/dist/**/*.js"],
"windows": {
"env": {
"DVCPATH": ".env\\Scripts\\dvc.exe",
"HOT_RELOAD": "true",
"PYTHONPATH": ".env\\Scripts\\python.exe",
"USE_DEV_UI": "true"
Expand All @@ -37,15 +35,13 @@
"${workspaceFolder}/demo"
],
"env": {
"DVCPATH": ".env/bin/dvc",
"HOT_RELOAD": "true",
"PYTHONPATH": ".env/bin/python3",
"USE_DEV_UI": "true"
},
"outFiles": ["${workspaceFolder}/extension/dist/**/*.js"],
"windows": {
"env": {
"DVCPATH": ".env\\Scripts\\dvc.exe",
"HOT_RELOAD": "true",
"PYTHONPATH": ".env\\Scripts\\python.exe",
"USE_DEV_UI": "true"
Expand Down
4 changes: 2 additions & 2 deletions extension/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -85,9 +85,9 @@
"title": "DVC",
"properties": {
"dvc.dvcPath": {
"title": "%config.dvcPath.title%",
"description": "%config.dvcPath.description%",
"type": "string",
"description": "Path to DVC.",
"default": "dvc",
"scope": "resource"
}
}
Expand Down
6 changes: 4 additions & 2 deletions extension/package.nls.json
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,8 @@
"command.initNoScm": "Init DVC --no-scm",
"command.push": "Push",
"command.runExperiment": "Run Experiment",
"command.selectDvcPath": "Select DVC Path",
"command.showExperiments": "Show Experiments"
"command.selectDvcPath": "Select DVC CLI Path",
"command.showExperiments": "Show Experiments",
"config.dvcPath.title": "DVC CLI Path",
"config.dvcPath.description": "Call DVC from this path. Follows Python Extension when blank."
}
88 changes: 42 additions & 46 deletions extension/src/Config.ts
Original file line number Diff line number Diff line change
Expand Up @@ -11,7 +11,7 @@ import {
import { Disposable } from '@hediet/std/disposable'
import { makeObservable, observable } from 'mobx'
import { WebviewColorTheme } from './webviews/experiments/contract'
import { findCliPath, findDvcRootPaths } from './fileSystem'
import { findDvcRootPaths } from './fileSystem'
import { Deferred } from '@hediet/std/synchronization'

export class Config {
Expand All @@ -21,7 +21,6 @@ export class Config {

public readonly dispose = Disposable.fn()
public readonly workspaceRoot: string
public dvcCliPath = 'dvc'
public dvcRootPaths: string[] = []

private onDidChangeEmitter: EventEmitter<ConfigurationChangeEvent>
Expand All @@ -48,17 +47,6 @@ export class Config {
this.dvcPathStatusBarItem.text = path
}

private setDvcPaths = async () => {
this.updateDvcPathStatusBarItem()
await this.setDvcCliPath()
return this.findDvcRoots()
}

private setDvcPathsOnActivation = async (dvcPath?: string) => {
await this.setDvcPath(dvcPath)
return this.setDvcPaths()
}

private getWorkspaceRoot = (): string => {
const { workspaceFolders } = workspace
if (!workspaceFolders || workspaceFolders.length === 0) {
Expand All @@ -68,18 +56,11 @@ export class Config {
return workspaceFolders[0].uri.fsPath
}

private setDvcCliPath = async (): Promise<void> => {
const path = await findCliPath(this.workspaceRoot, this.dvcPath)
if (path) {
this.dvcCliPath = path
}
}

public get dvcPath(): string {
return <string>workspace.getConfiguration().get('dvc.dvcPath')
return workspace.getConfiguration().get('dvc.dvcPath', '')
}

private setDvcPath(path = 'dvc'): Thenable<void> {
private setDvcPath(path?: string): Thenable<void> {
return workspace.getConfiguration().update('dvc.dvcPath', path)
}

Expand All @@ -88,35 +69,54 @@ export class Config {

dvcPathStatusBarItem.tooltip = 'Current DVC path.'
dvcPathStatusBarItem.command = 'dvc.selectDvcPath'
dvcPathStatusBarItem.text = this.dvcPath
dvcPathStatusBarItem.show()
return dvcPathStatusBarItem
}

public selectDvcPath = async (): Promise<string | undefined> => {
public selectDvcPath = async (): Promise<void> => {
const result = await window.showQuickPick(
[{ label: 'default' }, { label: 'custom' }],
{ placeHolder: 'Please choose...' }
[
{
label: 'Default',
description: 'Use Python Extension virtual environment if available',
picked: true,
value: undefined
},
{
label: 'Find',
description: 'Browse the filesystem for a DVC executable',
value: async () => {
const result = await window.showOpenDialog({
title: 'Select a DVC executable'
})
if (result) {
const [input] = result
const { fsPath } = input
this.setDvcPath(fsPath)
return fsPath
} else {
return undefined
}
}
}
],
{
placeHolder: 'Please choose...'
}
)
if (result) {
if (result.label === 'default') {
await this.setDvcPath()
return this.dvcPath
}
if (result.label === 'custom') {
const path = await window.showInputBox({
prompt: 'Enter a custom DVC path...'
})
await this.setDvcPath(path)
return this.dvcPath
const { value } = result
if (typeof value === 'function') {
await value()
} else {
this.setDvcPath(value)
}
}
}

private findDvcRoots = async () => {
const rootPaths = await findDvcRootPaths(
this.workspaceRoot,
this.dvcCliPath
)
const rootPaths = await findDvcRootPaths(this.workspaceRoot, this.dvcPath)
this.dvcRootPaths = rootPaths
}

Expand Down Expand Up @@ -147,14 +147,10 @@ export class Config {
)

this.dispose.track(
this.onDidChange(async () => {
this.setDvcPaths()
this.onDidChange(() => {
this.updateDvcPathStatusBarItem()
this.findDvcRoots()
})
)

const dvcOverridePath = process.env.DVCPATH
this.setDvcPathsOnActivation(dvcOverridePath).then(() =>
this._initialized.resolve()
)
}
}
25 changes: 4 additions & 21 deletions extension/src/IntegratedTerminal.ts
Original file line number Diff line number Diff line change
@@ -1,13 +1,8 @@
import { Extension, extensions, Terminal, window, workspace } from 'vscode'
import { Terminal, window, workspace } from 'vscode'
import { getRunExperimentCommand } from './cli/commands'
import { getReadyPythonExtension } from './extensions/python'
import { delay } from './util'

interface PythonExtensionAPI {
ready: Thenable<void>
}

type PythonExtension = Extension<PythonExtensionAPI>

// Static class that creates and holds a reference to an integrated terminal and can run commands in it.
export class IntegratedTerminal {
static termName = 'DVC'
Expand Down Expand Up @@ -37,12 +32,12 @@ export class IntegratedTerminal {
private static initializeInstance = async (): Promise<void> => {
IntegratedTerminal.deleteReferenceOnClose()

const pythonExtension = extensions.getExtension('ms-python.python')
const pythonExtension = await getReadyPythonExtension()
if (
pythonExtension &&
workspace.getConfiguration().get('python.terminal.activateEnvironment')
) {
return IntegratedTerminal.createPythonInstance(pythonExtension)
return IntegratedTerminal.createInstance(5000)
}

return IntegratedTerminal.createInstance(2000)
Expand All @@ -59,18 +54,6 @@ export class IntegratedTerminal {
})
}

private static createPythonInstance = async (
pythonExtension: PythonExtension
): Promise<void> => {
if (!pythonExtension.isActive) {
await IntegratedTerminal.waitForReadyPromise(pythonExtension)
}
return IntegratedTerminal.createInstance(5000)
}

private static waitForReadyPromise = async (extension: PythonExtension) =>
(await extension.activate()).ready

private static createInstance = async (ms: number): Promise<void> => {
IntegratedTerminal.instance = window.createTerminal({
name: IntegratedTerminal.termName
Expand Down
2 changes: 1 addition & 1 deletion extension/src/cli/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@ import { execCommand } from './reader'

export const add = async (options: {
fsPath: string
cliPath: string
cliPath: string | undefined
}): Promise<string> => {
const { fsPath, cliPath } = options

Expand Down
29 changes: 28 additions & 1 deletion extension/src/cli/reader.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -5,21 +5,48 @@ import {
initializeDirectory,
checkoutRecursive,
getRoot,
listDvcOnlyRecursive
listDvcOnlyRecursive,
getDvcInvocation
} from './reader'
import { execPromise } from '../util'
import complexExperimentsOutput from '../webviews/experiments/complex-output-example.json'
import { join, resolve } from 'path'
import { getPythonExecutionDetails } from '../extensions/python'

jest.mock('fs')
jest.mock('../util')
jest.mock('../extensions/python')

const mockedExecPromise = mocked(execPromise)
const mockedGetPythonExecutionDetails = mocked(getPythonExecutionDetails)

beforeEach(() => {
jest.resetAllMocks()
})

describe('getDvcInvocation', () => {
it('should utilize an interpreter path from the Python extension by default', async () => {
const testPythonBin = '/custom/path/to/python'
mockedGetPythonExecutionDetails.mockResolvedValue([testPythonBin])
expect(await getDvcInvocation({ cliPath: '', cwd: './' })).toEqual(
`${testPythonBin} -m dvc`
)
})

it('should ignore a path from the Python extension when cliPath is defined', async () => {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

[Q] Won't this mean that we won't be able to run experiments from the reader if cliPath is set? I.e we just bypass the virtual environment?

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Isn't the idea to remove the IntegratedTerminal as a next step?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Isn't the idea to remove the IntegratedTerminal as a next step?

That's the idea, as the lack of control we have over IntegratedTerminal warrants a rewrite that takes advantage of the Python extension introspection to make a shell instance of dvc that we can observe and control.

const testPythonBin = '/custom/path/to/python'
mockedGetPythonExecutionDetails.mockResolvedValue(['/wrong/python/bin'])
expect(
await getDvcInvocation({ cliPath: testPythonBin, cwd: './' })
).toEqual(testPythonBin)
})

it('should return a simple dvc call when no Python extension is present', async () => {
mockedGetPythonExecutionDetails.mockResolvedValue(undefined)
expect(await getDvcInvocation({ cliPath: '', cwd: './' })).toEqual('dvc')
})
})

describe('getExperiments', () => {
it('should match a snapshot when parsed', async () => {
const cwd = resolve()
Expand Down
20 changes: 15 additions & 5 deletions extension/src/cli/reader.ts
Original file line number Diff line number Diff line change
@@ -1,19 +1,29 @@
import { Commands } from './commands'
import { execPromise } from '../util'
import { ExperimentsRepoJSONOutput } from '../webviews/experiments/contract'
import { getPythonExecutionDetails } from '../extensions/python'

interface ReaderOptions {
cliPath: string
cliPath: string | undefined
cwd: string
}

export const execCommand = (
export const getDvcInvocation = async (options: ReaderOptions) => {
const { cliPath } = options
if (cliPath) {
return cliPath
}
const executionDetails = await getPythonExecutionDetails()
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

[Q] @rogermparent what are the actual execution details? just the path to python inside the virtual environment?

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

image

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

From here in the Python extension's src/client/api.ts

/**
  * E.g of execution commands returned could be,
  * * `['<path to the interpreter set in settings>']`
  * * `['<path to the interpreter selected by the extension when setting is not set>']`
  * * `['conda', 'run', 'python']` which is used to run from within Conda environments.
  * or something similar for some other Python environments.
  *
  * @type {(string[] | undefined)} When return value is `undefined`, it means no interpreter is set.
  * Otherwise, join the items returned using space to construct the full execution command.
  */

So yeah, a path to the current Python interpreter that we call with -m to use its binaries, or the conda run command to get an equivalent.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The above screenshot demonstrates that this won't be enough to run commands that need the environment to be activated.

return executionDetails ? `${executionDetails.join(' ')} -m dvc` : 'dvc'
}

export const execCommand = async (
options: ReaderOptions,
command: string
): Promise<{ stdout: string; stderr: string }> => {
const { cliPath, cwd } = options

return execPromise(`${cliPath} ${command}`, {
const { cwd } = options
const fullCommandString = `${await getDvcInvocation(options)} ${command}`
return execPromise(fullCommandString, {
cwd
})
}
Expand Down
Loading