Skip to content

Commit

Permalink
Add silence support
Browse files Browse the repository at this point in the history
With this patch there is a newly supported section in `.flowconfig`:

```ini
[silence]
/path/to/file.js
/path/to/dir
```

Paths in `[silence]` will still be typechecked, but any
errors will be silenced as if there was a `suppress_comment` comment string
on every line where there are errors.

I called it `silence` to make it stand out from `ignore` and `suppress_comment` as
it currently silences all errors, including lints.

I believe this fixes #869, though I admit I didn't read
the whole issue.
  • Loading branch information
LegNeato committed Oct 31, 2017
1 parent 74189a0 commit 55a773a
Show file tree
Hide file tree
Showing 23 changed files with 175 additions and 31 deletions.
30 changes: 25 additions & 5 deletions src/commands/commandUtils.ml
Original file line number Diff line number Diff line change
Expand Up @@ -300,18 +300,20 @@ type flowconfig_params = {
* can defer parsing of the lint settings until after the flowconfig lint settings
* are known, to properly detect redundant settings (and avoid false positives) *)
raw_lint_severities: string list;
silences: string list;
}

let list_of_string_arg = function
| None -> []
| Some arg_str -> Str.split (Str.regexp ",") arg_str

let collect_flowconfig_flags main ignores_str includes_str lib_str lints_str =
let collect_flowconfig_flags main ignores_str includes_str lib_str lints_str silence_str =
let ignores = list_of_string_arg ignores_str in
let includes = list_of_string_arg includes_str in
let libs = list_of_string_arg lib_str in
let raw_lint_severities = list_of_string_arg lints_str in
main { ignores; includes; libs; raw_lint_severities; }
let silences = list_of_string_arg silence_str in
main { ignores; includes; libs; raw_lint_severities; silences; }

let file_options =
let default_lib_dir ~no_flowlib tmp_dir =
Expand Down Expand Up @@ -341,6 +343,12 @@ let file_options =
Path_matcher.add acc path
) Path_matcher.empty paths
in
let silences_of_arg root paths =
List.fold_left (fun acc path ->
let path = Files.make_path_absolute root path in
Path_matcher.add acc path
) Path_matcher.empty paths
in
let lib_paths ~root flowconfig extras =
let flowtyped_path = Files.get_flowtyped_path root in
let has_explicit_flowtyped_lib = ref false in
Expand All @@ -365,7 +373,7 @@ let file_options =
| [] -> config_libs
| _ -> config_libs @ (List.map Path.make extras)
in
fun ~root ~no_flowlib ~temp_dir ~includes ~ignores ~libs flowconfig ->
fun ~root ~no_flowlib ~temp_dir ~includes ~ignores ~libs ~silences flowconfig ->
let default_lib_dir =
let no_flowlib = no_flowlib || FlowConfig.no_flowlib flowconfig in
Some (default_lib_dir ~no_flowlib temp_dir)
Expand All @@ -378,6 +386,10 @@ let file_options =
includes
|> List.rev_append (FlowConfig.includes flowconfig)
|> includes_of_arg root in
let silences =
silences
|> List.rev_append (FlowConfig.silences flowconfig)
|> silences_of_arg root in
{ Files.
default_lib_dir;
ignores;
Expand All @@ -386,6 +398,7 @@ let file_options =
module_file_exts = FlowConfig.module_file_exts flowconfig;
module_resource_exts = FlowConfig.module_resource_exts flowconfig;
node_resolver_dirnames = FlowConfig.node_resolver_dirnames flowconfig;
silences;
}

let ignore_flag prev = CommandSpec.ArgSpec.(
Expand All @@ -412,13 +425,20 @@ let lints_flag prev = CommandSpec.ArgSpec.(
~doc:"Specify one or more lint rules, comma separated"
)

let silence_flag prev = CommandSpec.ArgSpec.(
prev
|> flag "--silence" (optional string)
~doc:"Specify one or more files/directories to silence, comma separated"
)

let flowconfig_flags prev = CommandSpec.ArgSpec.(
prev
|> collect collect_flowconfig_flags
|> ignore_flag
|> include_flag
|> lib_flag
|> lints_flag
|> silence_flag
)

type command_params = {
Expand Down Expand Up @@ -589,8 +609,8 @@ let make_options ~flowconfig ~lazy_mode ~root (options_flags: Options_flags.t) =
let open Options_flags in
let file_options =
let no_flowlib = options_flags.no_flowlib in
let { includes; ignores; libs; raw_lint_severities=_; } = options_flags.flowconfig_flags in
file_options ~root ~no_flowlib ~temp_dir ~includes ~ignores ~libs flowconfig
let { includes; ignores; libs; raw_lint_severities=_; silences; } = options_flags.flowconfig_flags in
file_options ~root ~no_flowlib ~temp_dir ~includes ~ignores ~libs ~silences flowconfig
in
let lint_severities = parse_lints_flag
(FlowConfig.lint_severities flowconfig) options_flags.flowconfig_flags.raw_lint_severities
Expand Down
24 changes: 21 additions & 3 deletions src/commands/config/flowConfig.ml
Original file line number Diff line number Diff line change
Expand Up @@ -312,6 +312,8 @@ type config = {
strict_mode: StrictModeSettings.t;
(* config options *)
options: Opts.t;
(* non-root silence paths *)
silences: string list;
}

module Pp : sig
Expand Down Expand Up @@ -366,6 +368,9 @@ end = struct
(string_of_severity state)))
lint_severities

let silences o silences =
List.iter (fun silence -> (fprintf o "%s\n" silence)) silences

let strict o config =
let open Lints in
let strict_mode = config.strict_mode in
Expand All @@ -390,6 +395,9 @@ end = struct
section_header o "options";
options o config;
fprintf o "\n";
section_header o "silence";
silences o config.silences;
fprintf o "\n";
section_header o "strict";
strict o config
end
Expand All @@ -400,6 +408,7 @@ let empty_config = {
libs = [];
lint_severities = LintSettings.default_severities;
strict_mode = StrictModeSettings.empty;
silences = [];
options = Opts.default_options
}

Expand Down Expand Up @@ -445,6 +454,10 @@ let parse_ignores config lines =
let ignores = trim_lines lines in
{ config with ignores; }

let parse_silences config lines =
let silences = trim_lines lines in
{ config with silences; }

let parse_options config lines =
let open Opts in
let (>>=) = Core_result.(>>=) in
Expand Down Expand Up @@ -886,6 +899,7 @@ let parse_section config ((section_ln, section), lines) =
| "lints", _ -> parse_lints config lines
| "strict", _ -> parse_strict config lines
| "options", _ -> parse_options config lines
| "silence", _ -> parse_silences config lines
| "version", _ -> parse_version config lines
| _ -> error section_ln (spf "Unsupported config section: \"%s\"" section)

Expand All @@ -911,17 +925,19 @@ let read filename =
|> List.filter is_not_comment in
parse empty_config lines

let init ~ignores ~includes ~libs ~options ~lints =
let init ~ignores ~includes ~libs ~lints ~options ~silences =
let ignores_lines = List.map (fun s -> (1, s)) ignores in
let includes_lines = List.map (fun s -> (1, s)) includes in
let options_lines = List.map (fun s -> (1, s)) options in
let lib_lines = List.map (fun s -> (1, s)) libs in
let lint_lines = List.map (fun s -> (1, s)) lints in
let options_lines = List.map (fun s -> (1, s)) options in
let silences_lines = List.map (fun s -> (1, s)) silences in
let config = parse_ignores empty_config ignores_lines in
let config = parse_includes config includes_lines in
let config = parse_options config options_lines in
let config = parse_libs config lib_lines in
let config = parse_lints config lint_lines in
let config = parse_options config options_lines in
let config = parse_silences config silences_lines in
config

let write config oc = Pp.config oc config
Expand Down Expand Up @@ -950,6 +966,8 @@ let ignores config = config.ignores
let includes config = config.includes
(* library paths. no wildcards *)
let libs config = config.libs
(* silence paths. no wildcards *)
let silences config = config.silences

(* options *)
let all c = c.options.Opts.all
Expand Down
5 changes: 4 additions & 1 deletion src/commands/config/flowConfig.mli
Original file line number Diff line number Diff line change
Expand Up @@ -14,8 +14,9 @@ val init:
ignores: string list ->
includes: string list ->
libs: string list ->
options: string list ->
lints: string list ->
options: string list ->
silences: string list ->
config
val write: config -> out_channel -> unit

Expand All @@ -29,6 +30,8 @@ val ignores: config -> string list
val includes: config -> string list
(* library paths. no wildcards *)
val libs: config -> string list
(* non-root paths to silence *)
val silences: config -> string list

(* options *)
val all: config -> bool
Expand Down
3 changes: 2 additions & 1 deletion src/commands/configCommands.ml
Original file line number Diff line number Diff line change
Expand Up @@ -43,6 +43,7 @@ module Init = struct
let includes = flowconfig_flags.CommandUtils.includes in
let libs = flowconfig_flags.CommandUtils.libs in
let lints = flowconfig_flags.CommandUtils.raw_lint_severities in
let silences = flowconfig_flags.CommandUtils.silences in

let file = Server_files_js.config_file root in
if Sys.file_exists file
Expand All @@ -51,7 +52,7 @@ module Init = struct
FlowExitStatus.(exit ~msg Invalid_flowconfig)
end;

let config = FlowConfig.init ~ignores ~includes ~libs ~options ~lints in
let config = FlowConfig.init ~ignores ~includes ~libs ~lints ~options ~silences in

let out = Sys_utils.open_out_no_fail file in
FlowConfig.write config out;
Expand Down
2 changes: 1 addition & 1 deletion src/commands/genFlowFilesCommand.ml
Original file line number Diff line number Diff line change
Expand Up @@ -111,7 +111,7 @@ let main option_values root error_flags strip_root ignore_flag
FlowExitStatus.exit ~msg FlowExitStatus.Commandline_usage_error
);

let options = LsCommand.make_options ~root ~ignore_flag ~include_flag in
let options = LsCommand.make_options ~root ~ignore_flag ~include_flag ~silence_flag:None in
let _, libs = Files.init options in
let next_files =
LsCommand.get_ls_files ~root ~all:false ~options ~libs ~imaginary:false (Some src)
Expand Down
10 changes: 6 additions & 4 deletions src/commands/lsCommand.ml
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,7 @@ let spec = {
|> ignore_flag
|> include_flag
|> root_flag
|> silence_flag
|> json_flags
|> from_flag
|> flag "--all" no_arg
Expand Down Expand Up @@ -106,13 +107,14 @@ let rec iter_get_next ~f get_next =
List.iter f result;
iter_get_next ~f get_next

let make_options ~root ~ignore_flag ~include_flag =
let make_options ~root ~ignore_flag ~include_flag ~silence_flag =
let flowconfig = FlowConfig.get (Server_files_js.config_file root) in
let temp_dir = FlowConfig.temp_dir flowconfig in
let includes = CommandUtils.list_of_string_arg include_flag in
let ignores = CommandUtils.list_of_string_arg ignore_flag in
let libs = [] in
CommandUtils.file_options ~root ~no_flowlib:true ~temp_dir ~ignores ~includes ~libs flowconfig
let silences = CommandUtils.list_of_string_arg silence_flag in
CommandUtils.file_options ~root ~no_flowlib:true ~temp_dir ~ignores ~includes ~libs ~silences flowconfig

(* The problem with Files.wanted is that it says yes to everything except ignored files and libs.
* So implicitly ignored files (like files in another directory) pass the Files.wanted check *)
Expand Down Expand Up @@ -172,7 +174,7 @@ let get_next_append_const get_next const =
ret

let main
strip_root ignore_flag include_flag root_flag json pretty from all imaginary reason
strip_root ignore_flag include_flag root_flag silence_flag json pretty from all imaginary reason
input_file root_or_files () =

let files_or_dirs = get_filenames_from_input ~allow_imaginary:true input_file root_or_files in
Expand All @@ -192,7 +194,7 @@ let main
| _ -> None)
) in

let options = make_options ~root ~ignore_flag ~include_flag in
let options = make_options ~root ~ignore_flag ~include_flag ~silence_flag in
(* Turn on --no-flowlib by default, so that flow ls never reports flowlib files *)
let options = { options with Files.default_lib_dir = None; } in
let _, libs = Files.init options in
Expand Down
6 changes: 6 additions & 0 deletions src/common/files.ml
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,7 @@ type options = {
module_file_exts: SSet.t;
module_resource_exts: SSet.t;
node_resolver_dirnames: string list;
silences: Path_matcher.t;
}

let default_lib_dir options = options.default_lib_dir
Expand All @@ -24,6 +25,7 @@ let lib_paths options = options.lib_paths
let module_file_exts options = options.module_file_exts
let module_resource_exts options = options.module_resource_exts
let node_resolver_dirnames options = options.node_resolver_dirnames
let silences options = options.silences

let node_modules_containers = ref SSet.empty

Expand Down Expand Up @@ -293,6 +295,10 @@ let is_ignored (options: options) =
let is_included options f =
Path_matcher.matches options.includes f

(* true if a file path matches a [silence] path in config *)
let is_silenced options f =
Path_matcher.matches options.silences f

let wanted ~options lib_fileset =
let is_ignored_ = is_ignored options in
fun path -> not (is_ignored_ path) && not (SSet.mem path lib_fileset)
Expand Down
4 changes: 4 additions & 0 deletions src/common/files.mli
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,7 @@ type options = {
module_file_exts: SSet.t;
module_resource_exts: SSet.t;
node_resolver_dirnames: string list;
silences: Path_matcher.t;
}

val default_lib_dir: options -> Path.t option
Expand All @@ -24,6 +25,7 @@ val lib_paths: options -> Path.t list
val module_file_exts: options -> SSet.t
val module_resource_exts: options -> SSet.t
val node_resolver_dirnames: options -> string list
val silences: options -> Path_matcher.t

val node_modules_containers: SSet.t ref

Expand All @@ -40,6 +42,8 @@ val is_flow_file: options: options -> string -> bool
val is_ignored: options -> string -> bool
(* true if a file path matches an [include] path in config *)
val is_included: options -> string -> bool
(* true if a file path matches a [silence] path in config *)
val is_silenced: options -> string -> bool

val is_valid_path: options: options -> string -> bool

Expand Down
4 changes: 2 additions & 2 deletions src/flow_dot_js.ml
Original file line number Diff line number Diff line change
Expand Up @@ -55,7 +55,7 @@ let load_lib_files ~master_cx ~metadata files
match parse_content lib_file lib_content with
| ast, [] ->
let cx, syms = Type_inference_js.infer_lib_file
~metadata ~exclude_syms ~lint_severities:LintSettings.default_severities
~metadata ~exclude_syms ~file_options:None ~lint_severities:LintSettings.default_severities
lib_file ast
in

Expand Down Expand Up @@ -168,7 +168,7 @@ let infer_and_merge ~root filename ast =
let strict_mode = StrictModeSettings.empty in
let file_sigs = Utils_js.FilenameMap.singleton filename file_sig in
Merge_js.merge_component_strict
~metadata ~lint_severities ~strict_mode ~file_sigs
~metadata ~file_options:None ~lint_severities ~strict_mode ~file_sigs
~get_ast_unsafe:(fun _ -> ast)
~get_docblock_unsafe:(fun _ -> stub_docblock)
[filename] reqs [] master_cx
Expand Down
3 changes: 2 additions & 1 deletion src/services/inference/init_js.ml
Original file line number Diff line number Diff line change
Expand Up @@ -71,6 +71,7 @@ let load_lib_files ~master_cx ~options files =

let lib_file = File_key.LibFile file in
let lint_severities = options.Options.opt_lint_severities in
let file_options = Options.file_options options in
match parse_lib_file options file with
| Parsing.Parse_ok ast ->

Expand All @@ -82,7 +83,7 @@ let load_lib_files ~master_cx ~options files =
in

let cx, syms = Infer.infer_lib_file
~metadata ~exclude_syms ~lint_severities
~metadata ~exclude_syms ~file_options:(Some file_options) ~lint_severities
lib_file ast
in

Expand Down
6 changes: 4 additions & 2 deletions src/services/inference/merge_service.ml
Original file line number Diff line number Diff line change
Expand Up @@ -93,9 +93,10 @@ let merge_strict_context ~options component =

let metadata = Context.metadata_of_options options in
let lint_severities = Options.lint_severities options in
let file_options = Some (Options.file_options options) in
let strict_mode = Options.strict_mode options in
let cx = Merge_js.merge_component_strict
~metadata ~lint_severities ~strict_mode ~file_sigs
~metadata ~lint_severities ~file_options ~strict_mode ~file_sigs
~get_ast_unsafe:Parsing_service_js.get_ast_unsafe
~get_docblock_unsafe:Parsing_service_js.get_docblock_unsafe
~do_gc:(Options.is_debug_mode options)
Expand Down Expand Up @@ -135,9 +136,10 @@ let merge_contents_context options file ast info ~ensure_checked_dependencies =

let metadata = Context.metadata_of_options options in
let lint_severities = Options.lint_severities options in
let file_options = Some (Options.file_options options) in
let strict_mode = Options.strict_mode options in
let cx = Merge_js.merge_component_strict
~metadata ~lint_severities ~strict_mode ~file_sigs
~metadata ~file_options ~lint_severities ~strict_mode ~file_sigs
~get_ast_unsafe:(fun _ -> ast)
~get_docblock_unsafe:(fun _ -> info)
component file_reqs dep_cxs master_cx
Expand Down
Loading

0 comments on commit 55a773a

Please sign in to comment.