Compare commits
No commits in common. "e40c70be9cc405229d93682ec4056db1e0151ef8" and "2fd3da6ceb2fa10c6bf18ddaf189bb106099a4d4" have entirely different histories.
e40c70be9c
...
2fd3da6ceb
15 changed files with 39 additions and 269 deletions
|
@ -1,7 +0,0 @@
|
||||||
# v0.1.1 (2023-01-30)
|
|
||||||
|
|
||||||
- dune-project: lower dune requirement to 2.0
|
|
||||||
|
|
||||||
# v0.1.0 (2023-01-30)
|
|
||||||
|
|
||||||
- initial release
|
|
|
@ -1,3 +0,0 @@
|
||||||
Permission to use, copy, modify, and distribute this software for any purpose with or without fee is hereby granted, provided that the above copyright notice and this permission notice appear in all copies.
|
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
|
|
@ -1,3 +0,0 @@
|
||||||
# opam-graph
|
|
||||||
|
|
||||||
Visualizes dependencies of "opam switch export" as dot or svg.
|
|
2
app/dune
2
app/dune
|
@ -1,4 +1,4 @@
|
||||||
(executable
|
(executable
|
||||||
(name main)
|
(name main)
|
||||||
(public_name opam-graph)
|
(public_name opam_graph)
|
||||||
(libraries cmdliner logs logs.fmt fmt.cli fmt.tty logs.cli opam-graph))
|
(libraries cmdliner logs logs.fmt fmt.cli fmt.tty logs.cli opam-graph))
|
||||||
|
|
48
app/main.ml
48
app/main.ml
|
@ -12,39 +12,24 @@ let read_file file =
|
||||||
with _ -> invalid_arg ("Error opening file " ^ file)
|
with _ -> invalid_arg ("Error opening file " ^ file)
|
||||||
|
|
||||||
let jump () transitive file output_format =
|
let jump () transitive file output_format =
|
||||||
let module G = Opam_graph in
|
|
||||||
let switch = read_file file in
|
let switch = read_file file in
|
||||||
let data = OpamFile.SwitchExport.read_from_string switch in
|
let data = OpamFile.SwitchExport.read_from_string switch in
|
||||||
match output_format with
|
match output_format with
|
||||||
| `Text ->
|
| `Text ->
|
||||||
let graph = G.dependencies ~transitive data in
|
let graph = Opam_graph.dependencies ~transitive data in
|
||||||
Format.printf "%a" G.pp_graph graph
|
Format.printf "%a" Opam_graph.pp_graph graph
|
||||||
| `Dot ->
|
| `Dot ->
|
||||||
let graph = G.dependencies ~transitive data in
|
let graph = Opam_graph.dependencies ~transitive data in
|
||||||
let dot = G.Render.Dot.of_graph graph in
|
let dot = Opam_graph.Render.Dot.of_graph graph in
|
||||||
Format.printf "%a" G.Render.Dot.pp dot
|
Format.printf "%a" Opam_graph.Render.Dot.pp dot
|
||||||
| `Dot_ui ->
|
| `Dot_ui ->
|
||||||
let graph = G.Ui.dependencies ~transitive data in
|
let graph = Opam_graph.Ui.dependencies ~transitive data in
|
||||||
let dot = G.Render.Dot.of_assoc graph in
|
let dot = Opam_graph.Render.Dot.of_assoc graph in
|
||||||
Format.printf "%a" G.Render.Dot.pp dot
|
Format.printf "%a" Opam_graph.Render.Dot.pp dot
|
||||||
| `Html ->
|
| `Html ->
|
||||||
let graph = G.Ui.dependencies ~transitive data in
|
let graph = Opam_graph.Ui.dependencies ~transitive data in
|
||||||
let sharing_stats =
|
let html = Opam_graph.Render.Html.of_assoc graph in
|
||||||
data
|
Format.printf "%a" Opam_graph.Render.Html.pp html
|
||||||
|> G.dependencies ~transitive:false
|
|
||||||
|> G.calc_sharing_stats in
|
|
||||||
let override_css = "\
|
|
||||||
.deps-svg-wrap {\
|
|
||||||
background: rgb(60, 60, 87); \
|
|
||||||
}\
|
|
||||||
"
|
|
||||||
in
|
|
||||||
let html =
|
|
||||||
G.Render.Html.of_assoc
|
|
||||||
~override_css
|
|
||||||
~sharing_stats graph
|
|
||||||
in
|
|
||||||
Format.printf "%a" G.Render.Html.pp html
|
|
||||||
|
|
||||||
let setup_log style_renderer level =
|
let setup_log style_renderer level =
|
||||||
Fmt_tty.setup_std_outputs ?style_renderer ();
|
Fmt_tty.setup_std_outputs ?style_renderer ();
|
||||||
|
@ -77,15 +62,8 @@ let file =
|
||||||
Arg.(required & pos 0 (some file) None & info [ ] ~doc ~docv:"FILE")
|
Arg.(required & pos 0 (some file) None & info [ ] ~doc ~docv:"FILE")
|
||||||
|
|
||||||
let cmd =
|
let cmd =
|
||||||
let term = Term.(
|
let term = Term.(const jump $ setup_log $ transitive $ file $ output_format) in
|
||||||
const jump
|
let info = Cmd.info "opam_graph" ~version:"%%VERSION%%" in
|
||||||
$ setup_log
|
|
||||||
$ transitive
|
|
||||||
$ file
|
|
||||||
$ output_format
|
|
||||||
) in
|
|
||||||
let version = Fmt.str "%d" Opam_graph.visualization_version in
|
|
||||||
let info = Cmd.info "opam-graph" ~version in
|
|
||||||
Cmd.v info term
|
Cmd.v info term
|
||||||
|
|
||||||
let () = Cmd.eval cmd |> exit
|
let () = Cmd.eval cmd |> exit
|
||||||
|
|
|
@ -1,2 +1,2 @@
|
||||||
(lang dune 2.0)
|
(lang dune 2.6)
|
||||||
(name opam-graph)
|
(name opam-graph)
|
||||||
|
|
|
@ -1,9 +1,9 @@
|
||||||
opam-version: "2.0"
|
opam-version: "2.0"
|
||||||
maintainer: "Robur <team@robur.coop>"
|
maintainer: "Robur <team@robur.coop>"
|
||||||
authors: ["Robur <team@robur.coop>"]
|
authors: ["Robur <team@robur.coop>"]
|
||||||
homepage: "https://git.robur.coop/robur/opam-graph"
|
homepage: "https://git.robur.io/robur/opam-graph"
|
||||||
dev-repo: "git+https://git.robur.coop/robur/opam-graph.git"
|
dev-repo: "git+https://git.robur.io/robur/opam-graph.git"
|
||||||
bug-reports: "https://github.com/robur-coop/opam-graph/issues"
|
bug-reports: "https://github.com/roburio/opam-graph/issues"
|
||||||
license: "ISC"
|
license: "ISC"
|
||||||
|
|
||||||
depends: [
|
depends: [
|
||||||
|
@ -13,20 +13,18 @@ depends: [
|
||||||
"fmt" {>= "0.8.7"}
|
"fmt" {>= "0.8.7"}
|
||||||
"logs"
|
"logs"
|
||||||
"opam-core"
|
"opam-core"
|
||||||
"opam-format" {>= "2.1.1"}
|
"opam-format"
|
||||||
"ocamldot"
|
"ocamldot"
|
||||||
"tyxml" {>= "4.3.0"}
|
"rresult"
|
||||||
|
"tyxml"
|
||||||
"gg"
|
"gg"
|
||||||
]
|
]
|
||||||
build: [
|
build: [
|
||||||
["dune" "subst"] {dev}
|
["dune" "subst"] {dev}
|
||||||
["dune" "build" "-p" name "-j" jobs]
|
["dune" "build" "-p" name "-j" jobs]
|
||||||
["sh" "-ex" "packaging/FreeBSD/create_package.sh"] {os = "freebsd"}
|
|
||||||
["sh" "-ex" "packaging/debian/create_package.sh"] {os-family = "debian"}
|
|
||||||
]
|
]
|
||||||
|
|
||||||
synopsis: "Graphing dependencies of opam packages"
|
synopsis: "Graphing dependencies of opam packages"
|
||||||
description: """
|
description: """
|
||||||
This package outputs dependency graphs (in svg and dot) of opam package
|
This package outputs graphs (in svg and dot) of opam packages.
|
||||||
universes (opam switch export).
|
|
||||||
"""
|
"""
|
||||||
|
|
|
@ -1,15 +0,0 @@
|
||||||
name: opam-graph
|
|
||||||
version: %%VERSION_NUM%%
|
|
||||||
origin: local/opam-graph
|
|
||||||
comment: Opam graph visualization tool
|
|
||||||
www: https://git.robur.coop/robur/opam-graph
|
|
||||||
maintainer: Robur <team@robur.coop>
|
|
||||||
prefix: /usr/local
|
|
||||||
licenselogic: single
|
|
||||||
licenses: [ISCL]
|
|
||||||
flatsize: %%FLATSIZE%%
|
|
||||||
categories: [local]
|
|
||||||
desc = <<EOD
|
|
||||||
Graphing dependencies of opam packages in svg and dot
|
|
||||||
|
|
||||||
EOD;
|
|
|
@ -1,42 +0,0 @@
|
||||||
#!/bin/sh -e
|
|
||||||
|
|
||||||
# only execute anything if either
|
|
||||||
# - running under orb with package = opam-graph
|
|
||||||
# - not running under opam at all
|
|
||||||
if [ "$ORB_BUILDING_PACKAGE" != "opam-graph" -a "$OPAM_PACKAGE_NAME" != "" ]; then
|
|
||||||
exit 0;
|
|
||||||
fi
|
|
||||||
|
|
||||||
basedir=$(realpath "$(dirname "$0")"/../..)
|
|
||||||
pdir=$basedir/packaging/FreeBSD
|
|
||||||
bdir=$basedir/_build/install/default/bin
|
|
||||||
tmpd=$basedir/_build/stage
|
|
||||||
manifest=$tmpd/+MANIFEST
|
|
||||||
rootdir=$tmpd/rootdir
|
|
||||||
bindir=$rootdir/usr/local/bin
|
|
||||||
|
|
||||||
trap 'rm -rf $tmpd' 0 INT EXIT
|
|
||||||
|
|
||||||
mkdir -p "$bindir"
|
|
||||||
|
|
||||||
install -U "$bdir/opam-graph" "$bindir/opam-graph"
|
|
||||||
|
|
||||||
flatsize=$(find "$rootdir" -type f -exec stat -f %z {} + |
|
|
||||||
awk 'BEGIN {s=0} {s+=$1} END {print s}')
|
|
||||||
|
|
||||||
sed -e "s:%%FLATSIZE%%:${flatsize}:" -e "/^[Vv]ersion:/s/-/./g" "$pdir/MANIFEST" > "$manifest"
|
|
||||||
|
|
||||||
{
|
|
||||||
printf '\nfiles {\n'
|
|
||||||
find "$rootdir" -type f -exec sha256 -r {} + | sort |
|
|
||||||
awk '{print " " $2 ": \"" $1 "\","}'
|
|
||||||
fidn "$rootdir" -type l | sort |
|
|
||||||
awk '{print " " $1 ": -,"}'
|
|
||||||
printf '}\n'
|
|
||||||
} | sed -e "s:${rootdir}::" >> "$manifest"
|
|
||||||
|
|
||||||
export SOURCE_DATE_EPOCH=$(git log -1 --pretty=format:%ct)
|
|
||||||
pkg create -r "$rootdir" -M "$manifest" -o "$basedir/"
|
|
||||||
mv "$basedir"/opam-graph-*.pkg "$basedir/opam-graph.pkg"
|
|
||||||
echo 'bin: [ "opam-graph.pkg" ]' > "$basedir/opam-graph.install"
|
|
||||||
echo 'doc: [ "README.md" ]' >> "$basedir/opam-graph.install"
|
|
|
@ -1,5 +0,0 @@
|
||||||
opam-graph (%%VERSION_NUM%%) unstable; urgency=medium
|
|
||||||
|
|
||||||
* Initial release
|
|
||||||
|
|
||||||
--Robur team <team@robur.coop>
|
|
|
@ -1,12 +0,0 @@
|
||||||
Package: opam-graph
|
|
||||||
Version: %%VERSION_NUM%%
|
|
||||||
Section: unknown
|
|
||||||
Priority: optional
|
|
||||||
Maintainer: Robur Team <team@robur.coop>
|
|
||||||
Standards-Version: 4.4.1
|
|
||||||
Homepage: https://git.robur.coop/robur/opam-graph
|
|
||||||
Vcs-Browser: https://git.robur.coop/robur/opam-graph
|
|
||||||
Vcs-Git: https://git.robur.coop/robur/opam-graph.git
|
|
||||||
Architecture: FIXME
|
|
||||||
Description: Graphing dependencies of opam packages
|
|
||||||
This package outputs graphs (in svg and dot) of opam packages.
|
|
|
@ -1,8 +0,0 @@
|
||||||
Format: https://www.debian.org/doc/packaging-manuals/copyright-format/1.0/
|
|
||||||
Upstream-Name: opam-graph
|
|
||||||
Upstream-Contact: Robur Team <team@robur.coop>
|
|
||||||
Source: https://git.robur.coop/robur/opam-graph
|
|
||||||
|
|
||||||
Files: *
|
|
||||||
Copyright: "Robur Team <team@robur.coop>"
|
|
||||||
License: ISC
|
|
|
@ -1,32 +0,0 @@
|
||||||
#!/bin/sh -e
|
|
||||||
|
|
||||||
# only execute anything if either
|
|
||||||
# - running under orb with package = opam-graph
|
|
||||||
# - not running under opam at all
|
|
||||||
if [ "$ORB_BUILDING_PACKAGE" != "opam-graph" -a "$OPAM_PACKAGE_NAME" != "" ]; then
|
|
||||||
exit 0;
|
|
||||||
fi
|
|
||||||
|
|
||||||
basedir=$(realpath "$(dirname "$0")"/../..)
|
|
||||||
bdir=$basedir/_build/install/default/bin
|
|
||||||
tmpd=$basedir/_build/stage
|
|
||||||
rootdir=$tmpd/rootdir
|
|
||||||
bindir=$rootdir/usr/bin
|
|
||||||
debiandir=$rootdir/DEBIAN
|
|
||||||
|
|
||||||
trap 'rm -rf $tmpd' 0 INT EXIT
|
|
||||||
|
|
||||||
mkdir -p "$debiandir" "$bindir"
|
|
||||||
|
|
||||||
install "$bdir/opam-graph" "$bindir/opam-graph"
|
|
||||||
|
|
||||||
# install debian metadata
|
|
||||||
install -m 0644 $basedir/packaging/debian/control $debiandir/control
|
|
||||||
install -m 0644 $basedir/packaging/debian/changelog $debiandir/changelog
|
|
||||||
install -m 0644 $basedir/packaging/debian/copyright $debiandir/copyright
|
|
||||||
|
|
||||||
ARCH=$(dpkg-architecture -q DEB_TARGET_ARCH)
|
|
||||||
sed -i -e "s/^Architecture:.*/Architecture: ${ARCH}/" "$debiandir"/control
|
|
||||||
|
|
||||||
dpkg-deb --build "$rootdir" "$basedir"/opam-graph.deb
|
|
||||||
echo 'bin: [ "opam-graph.deb" ]' > "$basedir/opam-graph.install"
|
|
2
src/dune
2
src/dune
|
@ -1,5 +1,5 @@
|
||||||
(library
|
(library
|
||||||
(name opam_graph)
|
(name opam_graph)
|
||||||
(public_name opam-graph)
|
(public_name opam-graph)
|
||||||
(libraries opam-core opam-format dot tyxml gg)
|
(libraries opam-core opam-format dot rresult tyxml gg)
|
||||||
)
|
)
|
||||||
|
|
|
@ -1,16 +1,10 @@
|
||||||
|
|
||||||
let visualization_version = 1
|
|
||||||
(** Remember to increment this when anything changes that can affect the
|
|
||||||
visualization, e.g.:
|
|
||||||
* algorithm change
|
|
||||||
* UI change
|
|
||||||
* certain library-dependency changes
|
|
||||||
*)
|
|
||||||
|
|
||||||
let sprintf = Printf.sprintf
|
let sprintf = Printf.sprintf
|
||||||
|
|
||||||
module OSet = OpamPackage.Set
|
module OSet = OpamPackage.Set
|
||||||
|
|
||||||
|
type package = OpamPackage.t
|
||||||
|
|
||||||
let packages (switch : OpamFile.SwitchExport.t) =
|
let packages (switch : OpamFile.SwitchExport.t) =
|
||||||
assert (OSet.cardinal switch.selections.sel_pinned = 0);
|
assert (OSet.cardinal switch.selections.sel_pinned = 0);
|
||||||
assert (OSet.cardinal switch.selections.sel_compiler = 0);
|
assert (OSet.cardinal switch.selections.sel_compiler = 0);
|
||||||
|
@ -85,99 +79,26 @@ let pp_graph ppf graph =
|
||||||
(Name_set.elements deps))))
|
(Name_set.elements deps))))
|
||||||
graph.nodes
|
graph.nodes
|
||||||
|
|
||||||
let deps_of_opam =
|
|
||||||
let open OpamParserTypes.FullPos in
|
|
||||||
let ( let* ) = Result.bind in
|
|
||||||
let extract_pkg = function
|
|
||||||
| { pelem = String s ; _ } -> Ok (OpamPackage.Name.of_string s)
|
|
||||||
| _ -> Error (`Msg "expected a string")
|
|
||||||
in
|
|
||||||
let extract_list = function
|
|
||||||
| { pelem = List { pelem = deps ; _ } ; _ } ->
|
|
||||||
List.fold_left (fun acc d ->
|
|
||||||
let* acc = acc in
|
|
||||||
let* dep = extract_pkg d in
|
|
||||||
Ok (Name_set.add dep acc))
|
|
||||||
(Ok Name_set.empty) deps
|
|
||||||
| _ -> Error (`Msg "expected a list of strings")
|
|
||||||
in
|
|
||||||
let extract_deps = function
|
|
||||||
| { pelem = List { pelem = [ name ; deps ] ; _ } ; _ } ->
|
|
||||||
let* name = extract_pkg name in
|
|
||||||
let* deps = extract_list deps in
|
|
||||||
Ok (name, deps)
|
|
||||||
| { pelem = List { pelem = _lbody ; _ } ; _ } ->
|
|
||||||
Error (`Msg "expected exactly two strings")
|
|
||||||
| _ -> Error (`Msg "expected a pair of strings")
|
|
||||||
in
|
|
||||||
function
|
|
||||||
| { pelem = List { pelem = lbody ; _ } ; _ } ->
|
|
||||||
let* data =
|
|
||||||
List.fold_left (fun acc v ->
|
|
||||||
let* acc = acc in
|
|
||||||
let* deps = extract_deps v in
|
|
||||||
Ok (deps :: acc))
|
|
||||||
(Ok []) lbody
|
|
||||||
in
|
|
||||||
Ok (List.rev data)
|
|
||||||
| _ -> Error (`Msg "expected a list")
|
|
||||||
|
|
||||||
let retrieve_deps switch top =
|
|
||||||
let orb_deps = "x-orb-dependencies" in
|
|
||||||
let pkg_opam_file = opam_file switch top in
|
|
||||||
match OpamFile.OPAM.extended pkg_opam_file orb_deps deps_of_opam with
|
|
||||||
| None -> None
|
|
||||||
| Some Error `Msg _msg -> None
|
|
||||||
| Some Ok data ->
|
|
||||||
Some
|
|
||||||
(List.fold_left (fun acc (name, deps) ->
|
|
||||||
Name_map.add name deps acc)
|
|
||||||
Name_map.empty data)
|
|
||||||
|
|
||||||
let dependencies ~transitive (switch : OpamFile.SwitchExport.t) =
|
let dependencies ~transitive (switch : OpamFile.SwitchExport.t) =
|
||||||
let root_pkg = root switch in
|
let root_pkg = root switch in
|
||||||
let top = root_pkg.OpamPackage.name in
|
let top = root_pkg.OpamPackage.name in
|
||||||
let graph = { top ; nodes = Name_map.empty } in
|
let graph = { top ; nodes = Name_map.empty } in
|
||||||
let dep_map = retrieve_deps switch top in
|
|
||||||
let available = switch.selections.sel_installed in
|
let available = switch.selections.sel_installed in
|
||||||
let rec find_deps graph work =
|
let rec find_deps graph work =
|
||||||
match Name_set.choose_opt work with
|
match Name_set.choose_opt work with
|
||||||
| None -> graph
|
| None -> graph
|
||||||
| Some x ->
|
| Some x ->
|
||||||
|
let deps = match transitive with
|
||||||
|
| true -> transitive_dependencies switch x
|
||||||
|
| false -> direct_dependencies switch x
|
||||||
|
in
|
||||||
let deps =
|
let deps =
|
||||||
match dep_map with
|
deps
|
||||||
| None ->
|
|> Name_set.filter (fun name ->
|
||||||
let deps =
|
OpamPackage.Set.exists
|
||||||
match transitive with
|
(fun pkg -> pkg.OpamPackage.name = name)
|
||||||
| true -> transitive_dependencies switch x
|
available
|
||||||
| false -> direct_dependencies switch x
|
)
|
||||||
in
|
|
||||||
deps
|
|
||||||
|> Name_set.filter (fun name ->
|
|
||||||
OpamPackage.Set.exists
|
|
||||||
(fun pkg -> pkg.OpamPackage.name = name)
|
|
||||||
available
|
|
||||||
)
|
|
||||||
| Some map ->
|
|
||||||
let rec find_it seen work acc =
|
|
||||||
match Name_set.choose_opt work with
|
|
||||||
| None -> acc
|
|
||||||
| Some x ->
|
|
||||||
let seen = Name_set.add x seen
|
|
||||||
and work = Name_set.remove x work
|
|
||||||
in
|
|
||||||
match Name_map.find_opt x map with
|
|
||||||
| None -> find_it seen work acc
|
|
||||||
| Some deps ->
|
|
||||||
let work =
|
|
||||||
if transitive then
|
|
||||||
Name_set.union deps work
|
|
||||||
else
|
|
||||||
work
|
|
||||||
in
|
|
||||||
find_it seen work (Name_set.union deps acc)
|
|
||||||
in
|
|
||||||
find_it Name_set.empty (Name_set.singleton x) Name_set.empty
|
|
||||||
in
|
in
|
||||||
let graph = add_node graph x deps in
|
let graph = add_node graph x deps in
|
||||||
let work =
|
let work =
|
||||||
|
@ -224,7 +145,7 @@ module Ui = struct
|
||||||
|> Name_map.find root
|
|> Name_map.find root
|
||||||
in
|
in
|
||||||
let all_transitive_deps =
|
let all_transitive_deps =
|
||||||
if transitive = false then all_direct_deps else
|
if transitive = false then all_direct_deps else
|
||||||
dependencies ~transitive data
|
dependencies ~transitive data
|
||||||
in
|
in
|
||||||
let direct_deps_w_transitive_deps =
|
let direct_deps_w_transitive_deps =
|
||||||
|
@ -522,7 +443,7 @@ svg {
|
||||||
let make_direct_dep_edge_css dep =
|
let make_direct_dep_edge_css dep =
|
||||||
let dep = scoped_class dep in
|
let dep = scoped_class dep in
|
||||||
sprintf {|
|
sprintf {|
|
||||||
.deps-direct_dep.deps-edge.%s:hover ~
|
.deps-direct_dep.deps-edge.%s:hover ~
|
||||||
.deps-direct_dep.deps-node.%s {
|
.deps-direct_dep.deps-node.%s {
|
||||||
transform: scale(2);
|
transform: scale(2);
|
||||||
}
|
}
|
||||||
|
@ -768,7 +689,7 @@ svg {
|
||||||
let make_shared_deps_css_aux ~dep ~shared_deps =
|
let make_shared_deps_css_aux ~dep ~shared_deps =
|
||||||
shared_deps |> Seq.map (fun shared_dep ->
|
shared_deps |> Seq.map (fun shared_dep ->
|
||||||
sprintf {|
|
sprintf {|
|
||||||
.deps-direct_dep.%s:hover ~
|
.deps-direct_dep.%s:hover ~
|
||||||
.deps-node.deps-layer2_dep.%s {
|
.deps-node.deps-layer2_dep.%s {
|
||||||
fill: #5454ff;
|
fill: #5454ff;
|
||||||
filter: brightness(1.0) !important;
|
filter: brightness(1.0) !important;
|
||||||
|
@ -870,7 +791,7 @@ svg {
|
||||||
css :: acc
|
css :: acc
|
||||||
) sharing_stats []
|
) sharing_stats []
|
||||||
|> merge_css
|
|> merge_css
|
||||||
|
|
||||||
let of_assoc ~(sharing_stats:assoc_stats) (graph:assoc_graph) : _ output =
|
let of_assoc ~(sharing_stats:assoc_stats) (graph:assoc_graph) : _ output =
|
||||||
match graph with
|
match graph with
|
||||||
| [] -> { svg_content = []; svg_attr = []; css = "" }
|
| [] -> { svg_content = []; svg_attr = []; css = "" }
|
||||||
|
|
Loading…
Reference in a new issue