package datalog
 sectionYPositions = computeSectionYPositions($el), 10)"
  x-init="setTimeout(() => sectionYPositions = computeSectionYPositions($el), 10)"
  >
  
  
  An in-memory datalog implementation for OCaml
Install
    
    dune-project
 Dependency
Authors
Maintainers
Sources
  
    
      v0.6.tar.gz
    
    
        
    
  
  
  
    
  
  
    
  
        md5=4a2d12d630a5edd694968675a84a3ef5
    
    
  sha512=685c0e186705837cb3ac66df6e8011d9f6a9629484b3a813b767df95348d5a41f37301f3e199ed6c91a42a87d1563e8355377269176785b123eb297a5ad022d7
    
    
  doc/src/datalog.unix/Datalog_unix.ml.html
Source file Datalog_unix.ml
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177(* this file is part of datalog. See README for the license *) (** {1 Unix Interpreted Predicates} *) module TopDown = Datalog_top_down module type S = sig module TD : TopDown.S val setup_handlers : TD.DB.t -> unit end (* main constructor *) module Make(TD : TopDown.S) = struct module TD = TD module T = TD.T module C = TD.C let c2str = TD.Const.to_string let str2c = TD.Const.of_string let str_split ~by s = let rec next acc i j = if i = String.length s then if i > j then String.sub s j (i-j) :: acc else acc else if s.[i] = by then next (String.sub s j (i-j-1) :: acc) (i+1) (i+1) else next acc (i+1) j in next [] 0 0 (* list files in directory *) let _ls = function | T.Apply (head, [| T.Apply (path, [| |]) as lit1; _ |] ) -> begin try let l = ref [] in let path = String.trim (c2str path) in let d = Unix.opendir path in let rec next () = let cont = try let s = Unix.readdir d in l := s :: !l; true with End_of_file -> false in if cont then next () in next (); List.map (fun f -> let f = Filename.concat path f in let t = T.mk_apply head [| lit1; T.mk_const (str2c f) |] in C.mk_fact t) !l with _e -> [] end | _ -> [] let _stat lit = match lit with | T.Apply (head, [| T.Apply (path, [| |]) as t1; _ |]) -> begin try let path = String.trim (c2str path) in let stat = Unix.lstat path in let kind = match stat.Unix.st_kind with | Unix.S_LNK -> "link" | Unix.S_BLK -> "block" | Unix.S_DIR -> "dir" | Unix.S_FIFO -> "fifo" | Unix.S_REG -> "file" | Unix.S_SOCK -> "socket" | _ -> raise Exit in let kind = T.mk_const (str2c kind) in [ C.mk_fact (T.mk_apply head [| t1; kind |]) ] with Unix.Unix_error _ | Exit -> [] end | _ -> [] let _isDir lit = match lit with | T.Apply (_, [| T.Apply (path, [| |]) |]) -> begin try let path = String.trim (c2str path) in let stat = Unix.lstat path in let ok = match stat.Unix.st_kind with | Unix.S_DIR -> true | _ -> false in if ok then [ C.mk_fact lit ] else [] with Unix.Unix_error _ -> [] end | _ -> [] let _isFile lit = match lit with | T.Apply (_, [| T.Apply (path, [| |]) |]) -> begin try let path = String.trim (c2str path) in let stat = Unix.lstat path in let ok = match stat.Unix.st_kind with | Unix.S_REG -> true | _ -> false in if ok then [ C.mk_fact lit ] else [] with Unix.Unix_error _ -> [] end | _ -> [] let _isLink lit = match lit with | T.Apply (_, [| T.Apply (path, [| |]) |]) -> begin try let path = String.trim (c2str path) in let stat = Unix.lstat path in let ok = match stat.Unix.st_kind with | Unix.S_LNK -> true | _ -> false in if ok then [ C.mk_fact lit ] else [] with Unix.Unix_error _ -> [] end | _ -> [] let _getenv lit = match lit with | T.Apply (head, [| T.Apply (var, [| |]) as t1; _ |]) -> begin try let v = str2c (Unix.getenv (c2str var)) in let t = T.mk_apply head [| t1; T.mk_const v |] in [ C.mk_fact t ] with Not_found -> [] end | _ -> [] let _split lit = match lit with | T.Apply (head, [| T.Apply (s, [| |]) as t1; T.Apply (by, [| |]) as t2; _ |]) -> let by = c2str by in let s = c2str s in if String.length by = 1 then let l = str_split ~by:by.[0] s in List.map (fun tok -> C.mk_fact (T.mk_apply head [| t1; t2; T.mk_const (str2c tok) |])) l else [] | _ -> [] let _time = function | T.Apply (head, [| _ |]) -> let f = Unix.time () in let f = str2c (string_of_float f) in [ C.mk_fact (T.mk_apply head [| T.mk_const f |]) ] | _ -> [] let handlers = [ str2c "ls", "ls(path, file): lists files in path", _ls ; str2c "stat", "stat(path,X): binds X to the kind of file path is", _stat ; str2c "isDir", "isDir(path): true if the path is a directory", _isDir ; str2c "isFile", "isFile(path): true if the path is a proper file", _isFile ; str2c "isLink", "isLink(path): true if the path is a symbolic link", _isLink ; str2c "getenv", "getenv(v,X): binds X to the environment variable v", _getenv ; str2c "split", "split(str,by,X): binds X to sustrings of str delimited by by", _split ; str2c "time", "time(T): binds T to the current time", _time ] let setup_handlers db = TD.DB.interpret_list db handlers end module Default = Make(TopDown.Default)
 sectionYPositions = computeSectionYPositions($el), 10)"
  x-init="setTimeout(() => sectionYPositions = computeSectionYPositions($el), 10)"
  >