mirror of
https://github.com/c-cube/ocaml-containers.git
synced 2025-12-06 11:15:31 -05:00
375 lines
10 KiB
OCaml
375 lines
10 KiB
OCaml
module T = (val Containers_testlib.make ~__FILE__ ())
|
|
include T
|
|
open CCString
|
|
open Stdlib;;
|
|
|
|
q Q.printable_string (fun s -> s = rev (rev s));;
|
|
q Q.printable_string (fun s -> length s = length (rev s));;
|
|
q Q.printable_string (fun s -> rev s = (to_list s |> List.rev |> of_list));;
|
|
eq "abc" (rev "cba");;
|
|
eq "" (rev "");;
|
|
eq " " (rev " ")
|
|
|
|
let eq' = eq ~printer:string_of_int;;
|
|
|
|
eq' 1 (find ~sub:"bc" "abcd");;
|
|
eq' ~-1 (find ~sub:"bc" "abd");;
|
|
eq' 1 (find ~sub:"a" "_a_a_a_");;
|
|
eq' 6 (find ~start:5 ~sub:"a" "a1a234a");;
|
|
|
|
q ~count:10_000
|
|
Q.(pair printable_string printable_string)
|
|
(fun (s1, s2) ->
|
|
let i = find ~sub:s2 s1 in
|
|
i < 0 || String.sub s1 i (length s2) = s2)
|
|
|
|
let eq' = eq ~printer:Q.Print.(list int);;
|
|
|
|
eq' [ 1; 6 ] (find_all_l ~sub:"bc" "abc aabc aab");;
|
|
eq' [] (find_all_l ~sub:"bc" "abd");;
|
|
|
|
eq' [ 76 ]
|
|
(find_all_l ~sub:"aaaaaa"
|
|
"aabbaabbaaaaabbbbabababababbbbabbbabbaaababbbaaabaabbaabbaaaabbababaaaabbaabaaaaaabbbaaaabababaabaaabbaabaaaabbababbaabbaaabaabbabababbbaabababaaabaaababbbaaaabbbaabaaababbabaababbaabbaaaaabababbabaababbbaaabbabbabababaaaabaaababaaaaabbabbaabbabbbbbbbbbbbbbbaabbabbbbbabbaaabbabbbbabaaaaabbababbbaaaa")
|
|
;;
|
|
|
|
t @@ fun () -> mem ~sub:"bc" "abcd";;
|
|
t @@ fun () -> not (mem ~sub:"a b" "abcd")
|
|
|
|
let eq' = eq ~printer:string_of_int;;
|
|
|
|
eq' 1 (rfind ~sub:"bc" "abcd");;
|
|
eq' ~-1 (rfind ~sub:"bc" "abd");;
|
|
eq' 5 (rfind ~sub:"a" "_a_a_a_");;
|
|
eq' 4 (rfind ~sub:"bc" "abcdbcd");;
|
|
eq' 6 (rfind ~sub:"a" "a1a234a");;
|
|
|
|
q ~count:10_000
|
|
Q.(pair printable_string printable_string)
|
|
(fun (s1, s2) ->
|
|
let i = rfind ~sub:s2 s1 in
|
|
i < 0 || String.sub s1 i (length s2) = s2)
|
|
;;
|
|
|
|
eq ~printer:CCFun.id
|
|
(replace ~which:`All ~sub:"a" ~by:"b" "abcdabcd")
|
|
"bbcdbbcd"
|
|
;;
|
|
|
|
eq ~printer:CCFun.id
|
|
(replace ~which:`Left ~sub:"a" ~by:"b" "abcdabcd")
|
|
"bbcdabcd"
|
|
;;
|
|
|
|
eq ~printer:CCFun.id
|
|
(replace ~which:`Right ~sub:"a" ~by:"b" "abcdabcd")
|
|
"abcdbbcd"
|
|
;;
|
|
|
|
eq ~printer:CCFun.id
|
|
(replace ~which:`All ~sub:"ab" ~by:"hello" " abab cdabb a")
|
|
" hellohello cdhellob a"
|
|
;;
|
|
|
|
eq ~printer:CCFun.id
|
|
(replace ~which:`Left ~sub:"ab" ~by:"nope" " a b c d ")
|
|
" a b c d "
|
|
;;
|
|
|
|
eq ~printer:CCFun.id (replace ~sub:"a" ~by:"b" "1aa234a") "1bb234b";;
|
|
t @@ fun () -> Split.list_cpy ~by:"," "aa,bb,cc" = [ "aa"; "bb"; "cc" ];;
|
|
t @@ fun () -> Split.list_cpy ~by:"--" "a--b----c--" = [ "a"; "b"; ""; "c"; "" ]
|
|
;;
|
|
|
|
t @@ fun () ->
|
|
Split.list_cpy ~by:" " "hello world aie" = [ "hello"; ""; "world"; "aie" ]
|
|
;;
|
|
|
|
t @@ fun () -> Split.left ~by:" " "ab cde f g " = Some ("ab", "cde f g ");;
|
|
t @@ fun () -> Split.left ~by:"__" "a__c__e_f" = Some ("a", "c__e_f");;
|
|
t @@ fun () -> Split.left ~by:"_" "abcde" = None;;
|
|
t @@ fun () -> Split.left ~by:"bb" "abbc" = Some ("a", "c");;
|
|
t @@ fun () -> Split.left ~by:"a_" "abcde" = None;;
|
|
t @@ fun () -> Split.right ~by:" " "ab cde f g" = Some ("ab cde f", "g");;
|
|
t @@ fun () -> Split.right ~by:"__" "a__c__e_f" = Some ("a__c", "e_f");;
|
|
t @@ fun () -> Split.right ~by:"_" "abcde" = None;;
|
|
t @@ fun () -> Split.right ~by:"a_" "abcde" = None;;
|
|
|
|
eq
|
|
~printer:Q.Print.(list string)
|
|
[ "a"; "few"; "words"; "from"; "our"; "sponsors" ]
|
|
(split_on_char ' ' "a few words from our sponsors")
|
|
;;
|
|
|
|
q
|
|
Q.(printable_string)
|
|
(fun s ->
|
|
let s = split_on_char ' ' s |> String.concat " " in
|
|
s = (split_on_char ' ' s |> String.concat " "))
|
|
;;
|
|
|
|
t @@ fun () -> compare_versions "0.1.3" "0.1" > 0;;
|
|
t @@ fun () -> compare_versions "10.1" "2.0" > 0;;
|
|
t @@ fun () -> compare_versions "0.1.alpha" "0.1" > 0;;
|
|
t @@ fun () -> compare_versions "0.3.dev" "0.4" < 0;;
|
|
t @@ fun () -> compare_versions "0.foo" "0.0" < 0;;
|
|
t @@ fun () -> compare_versions "1.2.3.4" "01.2.4.3" < 0;;
|
|
|
|
q
|
|
Q.(pair printable_string printable_string)
|
|
(fun (a, b) ->
|
|
CCOrd.equiv (compare_versions a b) (CCOrd.opp compare_versions b a))
|
|
;;
|
|
|
|
t @@ fun () -> compare_natural "foo1" "foo2" < 0;;
|
|
t @@ fun () -> compare_natural "foo11" "foo2" > 0;;
|
|
t @@ fun () -> compare_natural "foo11" "foo11" = 0;;
|
|
t @@ fun () -> compare_natural "foo011" "foo11" = 0;;
|
|
t @@ fun () -> compare_natural "foo1a" "foo1b" < 0;;
|
|
t @@ fun () -> compare_natural "foo1a1" "foo1a2" < 0;;
|
|
t @@ fun () -> compare_natural "foo1a17" "foo1a2" > 0;;
|
|
|
|
q
|
|
Q.(pair printable_string printable_string)
|
|
(fun (a, b) -> CCOrd.opp compare_natural a b = compare_natural b a)
|
|
;;
|
|
|
|
q Q.(printable_string) (fun a -> compare_natural a a = 0);;
|
|
|
|
q
|
|
Q.(triple printable_string printable_string printable_string)
|
|
(fun (a, b, c) ->
|
|
if compare_natural a b < 0 && compare_natural b c < 0 then
|
|
compare_natural a c < 0
|
|
else
|
|
Q.assume_fail ())
|
|
;;
|
|
|
|
q Q.(string_of_size Gen.(0 -- 30)) (fun s -> edit_distance s s = 0);;
|
|
|
|
q
|
|
Q.(
|
|
let p = string_of_size Gen.(0 -- 20) in
|
|
pair p p)
|
|
(fun (s1, s2) -> edit_distance s1 s2 = edit_distance s2 s1)
|
|
;;
|
|
|
|
q
|
|
Q.(
|
|
let p = string_of_size Gen.(0 -- 20) in
|
|
pair p p)
|
|
(fun (s1, s2) ->
|
|
let e = edit_distance s1 s2 in
|
|
let e' = edit_distance ~cutoff:3 s1 s2 in
|
|
(if e' < 3 then
|
|
e = e'
|
|
else
|
|
e >= 3)
|
|
&&
|
|
if e <= 3 then
|
|
e = e'
|
|
else
|
|
true)
|
|
;;
|
|
|
|
eq ~printer:string_of_int 2 (edit_distance "hello" "helo!");;
|
|
eq ~printer:string_of_int 5 (edit_distance "abcde" "tuvwx");;
|
|
eq ~printer:string_of_int 2 (edit_distance ~cutoff:2 "abcde" "tuvwx");;
|
|
|
|
eq ~printer:string_of_int 1
|
|
(edit_distance ("a" ^ String.make 100 '_') ("b" ^ String.make 100 '_'))
|
|
;;
|
|
|
|
eq ~printer:string_of_int 1
|
|
(edit_distance ~cutoff:4
|
|
("a" ^ String.make 1000 '_')
|
|
("b" ^ String.make 1000 '_'))
|
|
;;
|
|
|
|
eq ~printer:string_of_int 2
|
|
(edit_distance ~cutoff:3
|
|
("a" ^ String.make 1000 '_' ^ "c")
|
|
("b" ^ String.make 1000 '_' ^ "d"))
|
|
;;
|
|
|
|
(* test that building a from s, and mutating one char of s, yields
|
|
a string s' that is accepted by a.
|
|
|
|
--> generate triples (s, i, c) where c is a char, s a non empty string
|
|
and i a valid index in s.
|
|
*)
|
|
|
|
q
|
|
(let gen =
|
|
Q.Gen.(
|
|
3 -- 10 >>= fun len ->
|
|
0 -- (len - 1) >>= fun i ->
|
|
string_size (return len) >>= fun s ->
|
|
char >|= fun c -> s, i, c)
|
|
in
|
|
let small (s, _, _) = String.length s in
|
|
Q.make ~small gen)
|
|
(fun (s, i, c) ->
|
|
let s' = Bytes.of_string s in
|
|
Bytes.set s' i c;
|
|
edit_distance s (Bytes.to_string s') <= 1)
|
|
;;
|
|
|
|
t @@ fun () -> prefix ~pre:"aab" "aabcd";;
|
|
t @@ fun () -> not (prefix ~pre:"ab" "aabcd");;
|
|
t @@ fun () -> not (prefix ~pre:"abcd" "abc");;
|
|
t @@ fun () -> prefix ~pre:"abc" "abcde";;
|
|
t @@ fun () -> prefix ~pre:"" "";;
|
|
t @@ fun () -> prefix ~pre:"" "abc";;
|
|
t @@ fun () -> prefix ~pre:"abc" "abc";;
|
|
t @@ fun () -> suffix ~suf:"cd" "abcd";;
|
|
t @@ fun () -> suffix ~suf:"" "";;
|
|
t @@ fun () -> suffix ~suf:"" "abc";;
|
|
t @@ fun () -> not (suffix ~suf:"cd" "abcde");;
|
|
t @@ fun () -> not (suffix ~suf:"abcd" "cd");;
|
|
eq ("ab", "cd") (take_drop 2 "abcd");;
|
|
eq ("abc", "") (take_drop 3 "abc");;
|
|
eq ("abc", "") (take_drop 5 "abc")
|
|
|
|
let eq' = eq ~printer:Q.Print.(option string);;
|
|
|
|
eq' (Some "ab") (chop_suffix ~suf:"cd" "abcd");;
|
|
eq' None (chop_suffix ~suf:"cd" "abcde");;
|
|
eq' None (chop_suffix ~suf:"abcd" "cd");;
|
|
eq' (Some "cd") (chop_prefix ~pre:"aab" "aabcd");;
|
|
eq' None (chop_prefix ~pre:"ab" "aabcd");;
|
|
eq' None (chop_prefix ~pre:"abcd" "abc")
|
|
|
|
let eq' = eq ~printer:Q.Print.string;;
|
|
|
|
eq' " 42" (pad 4 "42");;
|
|
eq' "0042" (pad ~c:'0' 4 "42");;
|
|
eq' "4200" (pad ~side:`Right ~c:'0' 4 "42");;
|
|
eq' "hello" (pad 4 "hello");;
|
|
eq' "aaa" (pad ~c:'a' 3 "");;
|
|
eq' "aaa" (pad ~side:`Right ~c:'a' 3 "");;
|
|
t @@ fun () -> of_list [ 'a'; 'b'; 'c' ] = "abc";;
|
|
t @@ fun () -> of_list [] = ""
|
|
|
|
let eq' = eq ~printer:Q.Print.(list @@ Printf.sprintf "%S");;
|
|
|
|
eq' [ "ab"; "c" ] (lines "ab\nc");;
|
|
eq' [ "ab"; "c" ] (lines "ab\nc\n");;
|
|
eq' [] (lines "");;
|
|
eq' [ "" ] (lines "\n");;
|
|
eq' [ ""; "a" ] (lines "\na");;
|
|
q Q.(printable_string) (fun s -> lines s = (lines_gen s |> Gen.to_list));;
|
|
q Q.(printable_string) (fun s -> lines s = (lines_iter s |> Iter.to_list));;
|
|
|
|
q
|
|
Q.(small_list printable_string)
|
|
(fun l -> concat_iter ~sep:"\n" (Iter.of_list l) = concat "\n" l)
|
|
;;
|
|
|
|
q
|
|
Q.(small_list printable_string)
|
|
(fun l -> concat_gen ~sep:"\n" (Gen.of_list l) = concat "\n" l)
|
|
;;
|
|
|
|
q
|
|
Q.(small_list printable_string)
|
|
(fun l -> concat_seq ~sep:"\n" (CCSeq.of_list l) = concat "\n" l)
|
|
;;
|
|
|
|
eq ~printer:CCFun.id "" (unlines []);;
|
|
eq ~printer:CCFun.id "ab\nc\n" (unlines [ "ab"; "c" ]);;
|
|
q Q.printable_string (fun s -> trim (unlines (lines s)) = trim s);;
|
|
q Q.printable_string (fun s -> trim (unlines_gen (lines_gen s)) = trim s);;
|
|
|
|
q
|
|
Q.(small_list small_string)
|
|
(fun l ->
|
|
let l = unlines l |> lines in
|
|
l = (unlines l |> lines))
|
|
;;
|
|
|
|
t @@ fun () -> set "abcd" 1 '_' = "a_cd";;
|
|
t @@ fun () -> set "abcd" 0 '-' = "-bcd";;
|
|
|
|
t @@ fun () ->
|
|
try
|
|
ignore (set "abc" 5 '_');
|
|
false
|
|
with Invalid_argument _ -> true
|
|
;;
|
|
|
|
eq ~printer:Q.Print.string "bcef"
|
|
(filter_map
|
|
(function
|
|
| 'c' -> None
|
|
| c -> Some (Char.chr (Char.code c + 1)))
|
|
"abcde")
|
|
;;
|
|
|
|
eq ~printer:Q.Print.string "abde"
|
|
(filter
|
|
(function
|
|
| 'c' -> false
|
|
| _ -> true)
|
|
"abcdec")
|
|
;;
|
|
|
|
q Q.printable_string (fun s -> filter (fun _ -> true) s = s);;
|
|
eq ~printer:Q.Print.string "abcde" (uniq Stdlib.( = ) "abbccdeeeee");;
|
|
eq ~printer:CCFun.id "abc " (ltrim " abc ");;
|
|
eq ~printer:CCFun.id " abc" (rtrim " abc ");;
|
|
q Q.(printable_string) (fun s -> String.trim s = (s |> ltrim |> rtrim));;
|
|
q Q.(printable_string) (fun s -> ltrim s = ltrim (ltrim s));;
|
|
q Q.(printable_string) (fun s -> rtrim s = rtrim (rtrim s));;
|
|
|
|
q
|
|
Q.(printable_string)
|
|
(fun s ->
|
|
let s' = ltrim s in
|
|
if s' = "" then
|
|
Q.assume_fail ()
|
|
else
|
|
s'.[0] <> ' ')
|
|
;;
|
|
|
|
q
|
|
Q.(printable_string)
|
|
(fun s ->
|
|
let s' = rtrim s in
|
|
if s' = "" then
|
|
Q.assume_fail ()
|
|
else
|
|
s'.[String.length s' - 1] <> ' ')
|
|
;;
|
|
|
|
t @@ fun () -> equal_caseless "foo" "FoO";;
|
|
t @@ fun () -> equal_caseless "helLo" "HEllO";;
|
|
|
|
q
|
|
Q.(pair printable_string printable_string)
|
|
(fun (s1, s2) ->
|
|
equal_caseless s1 s2 = (lowercase_ascii s1 = lowercase_ascii s2))
|
|
;;
|
|
|
|
q Q.(printable_string) (fun s -> equal_caseless s s);;
|
|
q Q.(printable_string) (fun s -> equal_caseless (uppercase_ascii s) s)
|
|
|
|
let eq' = eq ~printer:(Printf.sprintf "%S");;
|
|
|
|
eq' "0068656c6c6f20776f726c64" (to_hex "\000hello world");;
|
|
eq' "" (to_hex "");;
|
|
eq' "\000hello world" (of_hex_exn "0068656c6c6f20776f726c64");;
|
|
eq' "hello world" (of_hex_exn "68656C6C6F20776F726C64");;
|
|
q Q.(string) (fun s -> of_hex_exn (to_hex s) = s);;
|
|
|
|
q
|
|
Q.(string)
|
|
(fun s ->
|
|
CCString.for_all (function
|
|
| 'A' .. 'F' | 'a' .. 'f' | '0' .. '9' -> true
|
|
| _ -> false)
|
|
@@ to_hex s)
|
|
;;
|
|
|
|
t @@ fun () -> "ab" < "abc";;
|
|
t @@ fun () -> "123" < "14"
|