let main () =
let posargs = OptParse.OptParser.parse_argv Options.options in
let inputlist = posargs@(OptParse.Opt.get Options.foreground) in
let (input_format,implicit_format) = guess_format Options.inputtype inputlist in
Boilerplate.enable_debug (OptParse.Opt.get Options.verbose);
Boilerplate.enable_timers (OptParse.Opt.get Options.timers) ["Solver"];
Boilerplate.enable_bars (OptParse.Opt.get Options.progress)
["Depsolver_int.univcheck";"Depsolver_int.init_solver"] ;
Boilerplate.all_quiet (OptParse.Opt.get Options.quiet);
let options = Options.set_options input_format in
let fg = OptParse.Opt.get Options.foreground in
let bg = OptParse.Opt.get Options.background in
let fg =
let pos =
if List.length (posargs@fg@bg) = 0 && implicit_format then
["-"]
else
posargs
in
if implicit_format then add_format input_format (pos@fg) else (pos@fg)
in
let bg = if implicit_format then add_format input_format bg else bg in
let (preamble,pkgll,from_cudf,to_cudf) = Boilerplate.load_list ~options [fg;bg] in
let (fg_pkglist, bg_pkglist) = match pkgll with [fg;bg] -> (fg,bg) | _ -> assert false in
let fg_pkglist =
if OptParse.Opt.get Options.latest then
let h = Hashtbl.create (List.length fg_pkglist) in
List.iter (fun p ->
try
let q = Hashtbl.find h p.Cudf.package in
if (CudfAdd.compare p q) > 0 then
Hashtbl.replace h p.Cudf.package p
else ()
with Not_found -> Hashtbl.add h p.Cudf.package p
) fg_pkglist;
Hashtbl.fold (fun _ v acc -> v::acc) h []
else
fg_pkglist
in
let universe =
let s = CudfAdd.to_set (fg_pkglist @ bg_pkglist) in
Cudf.load_universe (CudfAdd.Cudf_set.elements s)
in
let universe_size = Cudf.universe_size universe in
if OptParse.Opt.is_set Options.checkonly &&
OptParse.Opt.is_set Options.coinst then
fatal "--checkonly and --coinst cannot be speficied together";
let checklist =
if OptParse.Opt.is_set Options.checkonly then begin
info "--checkonly specified, consider all packages as background packages";
List.flatten (
List.map (fun ((n,a),c) ->
let (name,filter) = Boilerplate.debvpkg to_cudf ((n,a),c) in
Cudf.lookup_packages ~filter universe name
) (OptParse.Opt.get Options.checkonly)
)
end else []
in
let coinstlist =
if OptParse.Opt.is_set Options.coinst then begin
info "--coinst specified, consider all packages as background packages";
List.map (fun ((n,a),c) ->
let (name,filter) = Boilerplate.debvpkg to_cudf ((n,a),c) in
Cudf.lookup_packages ~filter universe name
) (OptParse.Opt.get Options.coinst)
end else []
in
let pp ?(decode=CudfAdd.decode) pkg =
let (p,v) = from_cudf (pkg.Cudf.package,pkg.Cudf.version) in
let l =
List.filter_map (fun k ->
try Some(k,decode(Cudf.lookup_package_property pkg k))
with Not_found -> None
) ["architecture";"source";"sourcenumber";"essential"]
in (decode p,decode v,l)
in
info "Solving..." ;
let failure = OptParse.Opt.get Options.failures in
let success = OptParse.Opt.get Options.successes in
let explain = OptParse.Opt.get Options.explain in
let minimal = OptParse.Opt.get Options.minimal in
let summary = OptParse.Opt.get Options.summary in
let fmt =
if OptParse.Opt.is_set Options.outfile then
let oc = open_out (OptParse.Opt.get Options.outfile) in
Format.formatter_of_out_channel oc
else
Format.std_formatter
in
let results = Diagnostic.default_result universe_size in
if failure || success then Format.fprintf fmt "@[<v 1>report:@,";
let callback d =
if summary then Diagnostic.collect results d ;
let pp =
if input_format = Url.Cudf then
fun pkg -> pp ~decode:(fun x -> x) pkg
else fun pkg -> pp pkg
in
Diagnostic.fprintf ~pp ~failure ~success ~explain ~minimal fmt d
in
Util.Timer.start timer;
let number_broken =
if OptParse.Opt.is_set Options.coinst then
let rl = Depsolver.edos_coinstall_prod universe coinstlist in
let number_broken_tuples =
List.length (List.filter (fun r -> not (Diagnostic.is_solution r)) rl)
and number_checks = List.length rl
in begin
ignore(Util.Timer.stop timer ());
List.iter callback rl;
if failure || success then Format.fprintf fmt "@]@.";
Format.fprintf fmt "total-packages: %d@." universe_size;
Format.fprintf fmt "total-tuples: %d@." number_checks;
Format.fprintf fmt "broken-tuples: %d@." number_broken_tuples;
number_broken_tuples
end
else begin
let global_constraints = not(OptParse.Opt.get Options.deb_ignore_essential) in
let number_broken_packages =
if OptParse.Opt.is_set Options.checkonly then
Depsolver.listcheck ~global_constraints ~callback universe checklist
else begin
if bg_pkglist = [] then
Depsolver.univcheck ~global_constraints ~callback universe
else
Depsolver.listcheck ~global_constraints ~callback universe fg_pkglist
end
in
ignore(Util.Timer.stop timer ());
if failure || success then Format.fprintf fmt "@]@.";
let fn = List.length fg_pkglist in
let bn = List.length bg_pkglist in
let nb,nf =
let cl = List.length checklist in
if cl != 0 then ((fn + bn) - cl,cl) else (bn,fn)
in
Format.fprintf fmt "background-packages: %d@." nb;
Format.fprintf fmt "foreground-packages: %d@." nf;
Format.fprintf fmt "total-packages: %d@." universe_size;
Format.fprintf fmt "broken-packages: %d@." number_broken_packages;
if summary then
Format.fprintf fmt "@[%a@]@." (Diagnostic.pp_summary ~pp ()) results;
number_broken_packages
end
in
if number_broken > 0 then exit(1);