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

Experiment: Viper integration #78

Draft
wants to merge 100 commits into
base: master
Choose a base branch
from
Draft
Show file tree
Hide file tree
Changes from 9 commits
Commits
Show all changes
100 commits
Select commit Hold shift + click to select a range
0c99de9
Viper integration progress
rvanasa Oct 17, 2022
e796c87
Progress
rvanasa Oct 17, 2022
2b0ac87
Merge branch 'master' into viper
rvanasa Oct 17, 2022
5e67dd6
Add Viper as an extension dependency
rvanasa Oct 17, 2022
7ca8087
Merge branch 'viper' of https://github.com/dfinity/vscode-motoko into…
rvanasa Oct 17, 2022
fe96390
Add '-viper' suffix to extension version
rvanasa Oct 17, 2022
1076303
Optimize and clarify type checking
rvanasa Oct 17, 2022
50820c3
Add generated 'moc.js' file to gitignore
rvanasa Oct 17, 2022
cbab556
Add boilerplate for Viper LSP errors + source mapping
rvanasa Oct 20, 2022
8342646
Implement verification RPC calls
rvanasa Oct 20, 2022
3058bb3
Checkpoint: receiving diagnostics from StateChange notifications
rvanasa Oct 20, 2022
3cfc5bc
Add basic Motoko diagnostics from Viper LSP
rvanasa Oct 21, 2022
e18a93d
Add 'View in context' link from Motoko -> Viper diagnostics
rvanasa Oct 21, 2022
f519db6
Adjust diagnostic source messages
rvanasa Oct 21, 2022
aa0cfec
Update build script to account for z3 dependency
rvanasa Oct 21, 2022
f027fca
Misc
rvanasa Oct 21, 2022
2a519f2
Add 'implies' and 'invariant' to syntax highlighting
rvanasa Oct 21, 2022
fab6719
Fix initial setup process in package.json
rvanasa Oct 21, 2022
5c2b5cf
Fix
rvanasa Oct 21, 2022
494c2ab
// @viper -> // @verify
rvanasa Oct 21, 2022
ca5fd04
Only insert Viper code into '*.vpr' files
rvanasa Oct 21, 2022
31720e0
Rewrite verification error messages
rvanasa Oct 21, 2022
b98d69e
Improve unknown range handling
rvanasa Oct 21, 2022
3e5beed
Hide unused Viper LS message types
rvanasa Oct 21, 2022
4734df0
Add provisional success diagnostic
rvanasa Oct 21, 2022
12a4ea9
Change error message (lifting error information from Viper)
rvanasa Oct 22, 2022
be5321f
Misc fixes from code review
rvanasa Oct 22, 2022
2a82fa2
Merge branch 'viper' of https://github.com/dfinity/vscode-motoko into…
rvanasa Oct 22, 2022
0a490b3
Run 'npm audit fix'
rvanasa Oct 22, 2022
ca64a28
Fix verification success corner case
rvanasa Oct 23, 2022
36c88ef
tweaks (#80)
ggreif Oct 23, 2022
09ab3a8
Use '*.mo.vpr' file endings
rvanasa Oct 23, 2022
f3020e4
Improve verification success diagnostic
rvanasa Oct 23, 2022
2b3a960
Merge branch 'viper' of https://github.com/dfinity/vscode-motoko into…
rvanasa Oct 23, 2022
10d7fdf
Improve verification success detection
rvanasa Oct 23, 2022
6d38483
Switch RPC 'Verify' request to notification
rvanasa Oct 23, 2022
05863c4
Merge branch 'master' of https://github.com/dfinity/vscode-motoko int…
rvanasa Oct 23, 2022
95f22ca
Add icon for Motoko files
rvanasa Oct 23, 2022
4115385
Merge branch 'master' into viper
rvanasa Oct 23, 2022
4fab05b
Merge branch 'master' into viper
rvanasa Oct 23, 2022
dee8a27
remove premature invalidation
ggreif Oct 27, 2022
ea81dab
Disable workspace type-checking
rvanasa Oct 27, 2022
cad3795
LSP: wait till port is up and parse it (#86)
ggreif Nov 9, 2022
ca084b3
Look up Java home directory using viper-ide logic (#91)
rvanasa Nov 9, 2022
d7ee213
Merge branch 'master' of https://github.com/dfinity/vscode-motoko int…
rvanasa Nov 15, 2022
3ae85c4
Update marketplace details in package.json
rvanasa Nov 15, 2022
35ffeeb
Update readme for VS Marketplace
rvanasa Nov 15, 2022
c67e11c
Add TM for Formal Motoko (#87)
aterga Nov 15, 2022
2a01b69
Merge branch 'viper' of https://github.com/dfinity/vscode-motoko into…
rvanasa Nov 15, 2022
a514d30
Remove custom 'Formal Motoko' theme in favor of using light/dark them…
rvanasa Nov 15, 2022
6c82368
Update package-lock.json
rvanasa Nov 15, 2022
14f5805
Set 'MOC_UNLOCK_VERIFICATION' environment variable for moc.js
rvanasa Nov 15, 2022
867e624
some version bumps (#97)
ggreif Nov 18, 2022
f6f1bc2
chore: set up a `motoko` submodule (#98)
ggreif Nov 18, 2022
fec8d16
Replace tsc with esbuild bundler
rvanasa Nov 18, 2022
f26d1a2
Fix missing Wasm file for 'prettier-plugin-motoko'
rvanasa Nov 18, 2022
0eaea58
Fix .vscodeignore
rvanasa Nov 18, 2022
2b63a3c
simplify instructions
ggreif Nov 18, 2022
82bc022
Merge branch 'bundle-esbuild' into viper
rvanasa Nov 18, 2022
4445f11
Add missing dev-dependency
rvanasa Nov 18, 2022
1816b9e
Add 'node_modules' back into .vscodeignore
rvanasa Nov 18, 2022
d8e6e89
fix
ggreif Nov 18, 2022
7f7441e
Replace 'mkdir -p' with 'mkdirp' for Windows build support
rvanasa Nov 18, 2022
2062102
Merge branch 'master' of https://github.com/dfinity/vscode-motoko int…
rvanasa Nov 21, 2022
8ea25ba
Update readme
rvanasa Nov 21, 2022
b653fac
Add incompatibility prompt
rvanasa Nov 21, 2022
ecb80e0
Use custom syntax highlighting provided by Viper
rvanasa Nov 21, 2022
80e48c4
Add 'Background' section to readme
rvanasa Nov 21, 2022
b4a6caa
Temporarily rename extension to 'Formal Motoko'
rvanasa Nov 21, 2022
12b0854
Track debug launch configuration
rvanasa Nov 23, 2022
7419e39
First (#102)
aterga Nov 25, 2022
8779b86
Rename 'Formal Motoko' -> 'Motoko-san'
rvanasa Nov 25, 2022
b8548c4
chore: Viper is on Motoko `master` (#103)
ggreif Nov 26, 2022
dec28b3
v0.0.2
rvanasa Nov 25, 2022
008a6cd
Merge branch 'master' into viper
rvanasa Jan 4, 2023
c6b1eba
Misc. fixes
rvanasa Jan 4, 2023
e187a16
Merge remote-tracking branch 'origin/master' into viper
rvanasa Jan 4, 2023
aec67b0
Improve diagnostic behavior when opening/closing files
rvanasa Jan 4, 2023
015f176
Apply suggestions from code review
aterga Jan 4, 2023
b9d40af
Fix typo
aterga Jan 4, 2023
0f1180e
Update CI
rvanasa Jan 4, 2023
ed04f5b
Drop Node.js 14.x in CI
rvanasa Jan 4, 2023
b21008f
Deactivate test workflow for now (requires Nix installation)
rvanasa Jan 4, 2023
5dcd477
Explain CI blocker in comment
rvanasa Jan 4, 2023
62a59f8
track `motoko` ToT
ggreif Jan 5, 2023
acb9265
explain how to track the Motoko repo
ggreif Jan 5, 2023
d6d0279
tweak section
ggreif Jan 5, 2023
ccd6cc4
Bump 'vsce' to new package name
rvanasa Jan 5, 2023
dd5dd5b
Merge branch 'viper' of https://github.com/dfinity/vscode-motoko into…
rvanasa Jan 5, 2023
3c3b284
Merge
rvanasa Jan 5, 2023
d04f8fd
Tentatively re-enable test CI with Nix
rvanasa Jan 5, 2023
21ef478
Use Cachix action in CI (without auth token)
rvanasa Jan 5, 2023
ba913e1
Deactivate CI (#145)
rvanasa Jan 6, 2023
563e565
Bump dependencies
rvanasa Mar 17, 2023
0e5c9fe
Run formatter; organize imports
rvanasa Apr 10, 2023
9639874
Fix 'Restart language server' command
rvanasa Apr 10, 2023
18e50a2
0.0.3
rvanasa Apr 10, 2023
7f38803
Refactor
rvanasa Apr 10, 2023
9dbb92e
Add support for WSL (#198)
rvanasa Apr 11, 2023
e45ff0d
Draft: Fix minor issues to enable Serokell to use the IDE for local t…
aterga Jul 16, 2024
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
2 changes: 2 additions & 0 deletions .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -2,3 +2,5 @@
/out/
result
*.vsix

/src/generated/**/*
18 changes: 0 additions & 18 deletions .vscode/launch.json

This file was deleted.

934 changes: 932 additions & 2 deletions package-lock.json

Large diffs are not rendered by default.

11 changes: 8 additions & 3 deletions package.json
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@
"name": "vscode-motoko",
"displayName": "Motoko",
"description": "Motoko language support",
"version": "0.5.3",
"version": "0.5.3-viper",
"publisher": "dfinity-foundation",
"repository": "https://github.com/dfinity/vscode-motoko",
"engines": {
Expand Down Expand Up @@ -155,9 +155,14 @@
"typescript": "^4.8.2",
"vsce": "^2.11.0"
},
"extensionDependencies": [
"viper-admin.viper"
Copy link
Collaborator

Choose a reason for hiding this comment

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

Note: there might be a conflict between the version of viperserver provided by the viper-admin.viper extension and Nix. Maybe we don't need this for now and just assume that viperserver.jar is available in the system (e.g., under some env variable that Nix provides)?

Copy link
Collaborator

Choose a reason for hiding this comment

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

Copy link
Contributor

Choose a reason for hiding this comment

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

He already uses $VIPER_SERVER! But the ideal thing would be to frob the port number form the viper-ide panel, and connect to the already running server.

Copy link
Contributor Author

@rvanasa rvanasa Oct 22, 2022

Choose a reason for hiding this comment

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

This would be ideal. The main challenge here is that VS Code loads extensions based on activation events, so it's currently required to manually open a *.vpr file to activate the Viper LS.

Update: I just found a way to programmatically activate the Viper extension using the extensions API, so now we just need a way to reliably retrieve the port number.

Looking at the corresponding logic in viper-ide, one option is to modify the extension to expose the port in the ViperApi (which we can then access from our language client). This should be sufficiently decoupled from our use case to make this a reasonable standalone feature.

],
"scripts": {
"vscode:prepublish": "npm run compile",
"compile": "rimraf ./out && tsc -p .",
"vscode:prepublish": "npm run generate && npm run compile",
"generate": "npm run download && cd ../motoko && nix-shell --command 'make -C src moc.js' && cp -vL src/moc.js ../vscode-motoko/src/generated",
"download": "curl -L -o src/generated/viperserver.jar https://github.com/viperproject/viperserver/releases/download/v-2022-10-18-0728/viperserver.jar -C -",
Copy link
Contributor

Choose a reason for hiding this comment

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

You can use $VIPER_SERVER in the first place.

Copy link
Contributor Author

@rvanasa rvanasa Oct 21, 2022

Choose a reason for hiding this comment

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

Good idea; now using the Nix environment for the stand-in portable build logic.

"compile": "rimraf ./out && tsc -p . && cp -vr src/generated out/generated",
"test": "jest",
"lint": "tslint -p .",
"package": "vsce package",
Expand Down
2 changes: 1 addition & 1 deletion src/server/ast.ts
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
import { AST } from 'motoko/lib/ast';
import { resolveVirtualPath, tryGetFileText } from './utils';
import { fromAST, Program } from './syntax';
import mo from 'motoko';
import mo from './motoko';

export interface AstStatus {
uri: string;
Expand Down
5 changes: 5 additions & 0 deletions src/server/motoko.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
import getMotoko from 'motoko/lib';

const mo = getMotoko(require('../generated/moc.js').Motoko as any);

export default mo;
40 changes: 37 additions & 3 deletions src/server/server.ts
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
import { execSync } from 'child_process';
import * as glob from 'fast-glob';
import { existsSync, readFileSync } from 'fs';
import mo from 'motoko';
import { existsSync, readFileSync, unlinkSync, writeFileSync } from 'fs';
import mo from './motoko';
import { Node } from 'motoko/lib/ast';
import { keywords } from 'motoko/lib/keywords';
import * as baseLibrary from 'motoko/packages/latest/base.json';
Expand Down Expand Up @@ -41,6 +41,7 @@ import {
resolveFilePath,
resolveVirtualPath,
} from './utils';
import { compileViper, invalidateViper } from './viper';

interface Settings {
motoko: MotokoSettings;
Expand Down Expand Up @@ -500,7 +501,38 @@ function check(uri: string | TextDocument): boolean {
}

console.log('~', virtualPath);
let diagnostics = mo.check(virtualPath) as any as Diagnostic[];

let diagnostics: Diagnostic[] = [];
let regularTypeCheck = true;

// Check for a `// @viper` comment at the beginning of the Motoko file
if (/^\s*\/\/ *@viper/i.test(getFileText(resolvedUri))) {
const viperFile = resolveFilePath(
resolvedUri.replace(/\.mo$/, '.vpr'),
);
try {
// @ts-ignore
const result = compileViper(resolvedUri, virtualPath);
console.log('Viper file:', viperFile);
if (result.diagnostics) {
diagnostics = result.diagnostics;
}
if (result.code) {
const [source, _lookup] = result.code;
writeFileSync(viperFile, source, 'utf8');
} else if (existsSync(viperFile)) {
unlinkSync(viperFile);
}
regularTypeCheck = false; // Already type checked in `compileViper()`
} catch (err) {
console.error(`Error while translating to Viper: ${err}`);
writeFileSync(viperFile, err?.toString() || '', 'utf8');
}
}

if (regularTypeCheck) {
diagnostics = mo.check(virtualPath) as any as Diagnostic[];
}

if (settings) {
if (settings.maxNumberOfProblems > 0) {
Expand Down Expand Up @@ -593,10 +625,12 @@ function writeVirtual(path: string, content: string) {
// content = preprocessMotoko(content);
// }
mo.write(path, content);
invalidateViper(path);
}

function deleteVirtual(path: string) {
mo.delete(path);
invalidateViper(path);
}

connection.onCodeAction((event) => {
Expand Down
86 changes: 86 additions & 0 deletions src/server/viper.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,86 @@
// Load custom `moc.js` with Viper integration
import { Span } from 'motoko/lib/ast';
import mo from './motoko';
import { spawn } from 'child_process';
import * as rpc from 'vscode-jsonrpc/node';
import { resolve } from 'path';
import { connect } from 'net';

const serverPort = 54816; // TODO: config

let connection: rpc.MessageConnection | undefined;

try {
spawn('java', [
'-Xmx2048m',
'-Xss16m',
'-jar',
resolve(__dirname, '../generated/viperserver.jar'),
'--singleClient',
'--serverMode',
'LSP',
'--port',
String(serverPort),
]).on('error', console.error);

const socket = connect(serverPort);
connection = rpc.createMessageConnection(
new rpc.SocketMessageReader(socket),
new rpc.SocketMessageWriter(socket),
);
connection.listen();

console.log('Listening to Viper LSP');

connection.sendNotification(new rpc.NotificationType('initialize'));
} catch (err) {
console.error(`Error while initializing Viper LSP: ${err}`);
}

type Range = { start: Span; end: Span };
type CompilerRange = [number, number, number, number];
export type Lookup = (pos: CompilerRange) => CompilerRange;

// Viper -> Motoko source map cache
const sourceLookupMap = new Map<string, Lookup>();

export function getMotokoSourceRange(
virtualPath: string,
{ start: [a, b], end: [c, d] }: Range,
): Range | undefined {
const lookup = sourceLookupMap.get(virtualPath);
if (!lookup) {
return;
}
const result = lookup([a, b, c, d]);
if (!result) {
return;
}
return {
start: [result[0], result[1]],
end: [result[2], result[3]],
};
}

export function compileViper(uri: string, virtualPath: string) {
const result = mo.compiler.viper([virtualPath]);
const lookup = result?.code?.[1];
if (lookup) {
sourceLookupMap.set(virtualPath, lookup);
}
if (connection) {
// TODO
connection.sendNotification(
new rpc.NotificationType('textDocument/didOpen'),
{
textDocument: { languageId: 'viper', version: 0, uri },
},
);
}

return result;
}

export function invalidateViper(virtualPath: string) {
sourceLookupMap.delete(virtualPath);
}