Skip to content

Commit

Permalink
Separate data and control in the interactive protocol
Browse files Browse the repository at this point in the history
Apart from using JSON messages, the new protocol is essentially the same as the
old one.  Significant differences:

- Instead of crashing, invalid requests (missing fields, unsupported queries,
  etc) now cause F* to print a reasonably helpful diagnostic message.
- Each query now includes an editor-supplied query id, and each response
  contains the id of the corresponding query.
- Some of the queries were renamed (info → lookup, completions → autocomplete)
  for consistency (all query names are now verbs).
- Messages that F* used to print to stdout or stderr are now translated into
  proper JSON messages and labeled with a severity level (a review of these
  messages is needed to decide which of these should be turned into properly
  reported errors or warnings including range information).
- F* now prints the protocol's version and a list of supported interactive
  features before starting the interactive loop.  This lets the Emacs mode do
  dynamic capability detection instead of relying on version numbers. The same
  information is available through a new "describe-protocol" query.
- A failed “push” doesn't require the editor to immediately send a corresponding
  “pop” any more — instead, the pop happens automatically.

Fixes #361, #366, and #828.
  • Loading branch information
cpitclaudel committed Apr 12, 2017
1 parent 9c81162 commit 90e901f
Show file tree
Hide file tree
Showing 8 changed files with 472 additions and 359 deletions.
11 changes: 10 additions & 1 deletion src/basic/FStar.Options.fs
Original file line number Diff line number Diff line change
Expand Up @@ -112,6 +112,7 @@ let init () =
("hide_uvar_nums" , Bool false);
("hint_info" , Bool false);
("in" , Bool false);
("ide" , Bool false);
("include" , List []);
("indent" , Bool false);
("initial_fuel" , Int 2);
Expand Down Expand Up @@ -203,6 +204,7 @@ let get_hide_genident_nums () = lookup_opt "hide_genident_nums"
let get_hide_uvar_nums () = lookup_opt "hide_uvar_nums" as_bool
let get_hint_info () = lookup_opt "hint_info" as_bool
let get_in () = lookup_opt "in" as_bool
let get_ide () = lookup_opt "ide" as_bool
let get_include () = lookup_opt "include" (as_list as_string)
let get_indent () = lookup_opt "indent" as_bool
let get_initial_fuel () = lookup_opt "initial_fuel" as_int
Expand Down Expand Up @@ -447,6 +449,11 @@ let rec specs () : list<Getopt.opt> =
ZeroArgs (fun () -> Bool true),
"Legacy interactive mode; reads input from stdin");

( noshort,
"ide",
ZeroArgs (fun () -> Bool true),
"JSON-based interactive mode for IDEs");

( noshort,
"include",
OneArg ((fun s -> List (List.map String (get_include()) @ [Path s])),
Expand Down Expand Up @@ -928,12 +935,14 @@ let full_context_dependency () = true
let hide_genident_nums () = get_hide_genident_nums ()
let hide_uvar_nums () = get_hide_uvar_nums ()
let hint_info () = get_hint_info ()
let ide () = get_ide ()
let indent () = get_indent ()
let initial_fuel () = min (get_initial_fuel ()) (get_max_fuel ())
let initial_ifuel () = min (get_initial_ifuel ()) (get_max_ifuel ())
let inline_arith () = get_inline_arith ()
let interactive () = get_in ()
let interactive () = get_in () || get_ide ()
let lax () = get_lax ()
let legacy_interactive () = get_in ()
let log_queries () = get_log_queries ()
let log_types () = get_log_types ()
let max_fuel () = get_max_fuel ()
Expand Down
2 changes: 2 additions & 0 deletions src/basic/FStar.Options.fsi
Original file line number Diff line number Diff line change
Expand Up @@ -78,13 +78,15 @@ val full_context_dependency : unit -> bool
val hide_genident_nums : unit -> bool
val hide_uvar_nums : unit -> bool
val hint_info : unit -> bool
val ide : unit -> bool
val include_path : unit -> list<string>
val indent : unit -> bool
val initial_fuel : unit -> int
val initial_ifuel : unit -> int
val inline_arith : unit -> bool
val interactive : unit -> bool
val lax : unit -> bool
val legacy_interactive : unit -> bool
val log_queries : unit -> bool
val log_types : unit -> bool
val max_fuel : unit -> int
Expand Down
3 changes: 3 additions & 0 deletions src/basic/ml/FStar_Util.ml
Original file line number Diff line number Diff line change
Expand Up @@ -359,6 +359,7 @@ let default_printer =
let current_printer = ref default_printer
let set_printer printer = current_printer := printer

let print_raw s = pr "%s" s; flush stdout
let print_string s = (!current_printer).printer_prinfo s
let print_any s = (!current_printer).printer_prinfo (Marshal.to_string s [])
let strcat s1 s2 = s1 ^ s2
Expand Down Expand Up @@ -528,6 +529,8 @@ let rec find_map l f =
| None -> find_map tl f
| y -> y

let try_find f l = try Some (List.find f l) with Not_found -> None

let try_find_index f l =
let rec aux i = function
| [] -> None
Expand Down
3 changes: 3 additions & 0 deletions src/basic/util.fs
Original file line number Diff line number Diff line change
Expand Up @@ -341,6 +341,7 @@ let default_printer =
let current_printer = ref default_printer
let set_printer printer = current_printer := printer

let print_raw s = pr "%s" s
let print_string s = (!current_printer).printer_prinfo s
let print_any s = print_string (spr "%A" s)
let strcat s1 s2 = s1 ^ s2
Expand Down Expand Up @@ -477,6 +478,8 @@ let find_opt f l =
| hd::tl -> if f hd then Some hd else aux tl in
aux l

let try_find f l = List.tryFind f l

let try_find_index f l = List.tryFindIndex f l

let sort_with f l = List.sortWith f l
Expand Down
2 changes: 2 additions & 0 deletions src/basic/util.fsi
Original file line number Diff line number Diff line change
Expand Up @@ -139,6 +139,7 @@ type printer = {
val default_printer : printer
val set_printer : printer -> unit

val print_raw : string -> unit
val print_string : string -> unit
val print_any : 'a -> unit
val strcat : string -> string -> string
Expand Down Expand Up @@ -237,6 +238,7 @@ val sort_with: ('a -> 'a -> int) -> list<'a> -> list<'a>
val set_eq: ('a -> 'a -> int) -> list<'a> -> list<'a> -> bool
val remove_dups: ('a -> 'a -> bool) -> list<'a> -> list<'a>
val add_unique: ('a -> 'a -> bool) -> 'a -> list<'a> -> list<'a>
val try_find: ('a -> bool) -> list<'a> -> option<'a>
val try_find_i: (int -> 'a -> bool) -> list<'a> -> option<(int * 'a)>
val find_map: list<'a> -> ('a -> option<'b>) -> option<'b>
val try_find_index: ('a -> bool) -> list<'a> -> option<int>
Expand Down
Loading

0 comments on commit 90e901f

Please sign in to comment.