open Biocaml_internal_pervasives
module Roman = struct
exception BadNumeral of char
let eval = function
| 'I' -> 1
| 'V' -> 5
| 'X' -> 10
| 'L' -> 50
| 'C' -> 100
| 'D' -> 500
| 'M' -> 1000
| c -> raise(BadNumeral(c))
let arabic (x:string) : int =
let n = String.length x in
let rec loop i v p =
if i < n then
let c = eval (String.get x i) in
loop (i+1) (if p < c then v-p else v+p) c
else v + p
in
if n > 0
then let v = eval (String.get x 0) in loop 1 0 v
else 0
let numerals_size = 7
let digit_size = function
| 0 -> 0 | 1 -> 1 | 2 -> 2 | 3 -> 3 | 4 -> 2
| 5 -> 1 | 6 -> 2 | 7 -> 3 | 8 -> 4 | 9 -> 2
| _ -> assert false
let rec count_size k = function
| (0,_) -> k
| (n,j) ->
if j >= 3
then count_size (digit_size (n mod 10) + k) (n/10,j-2)
else n+k
let roman_size n =
if n < 0 then 0 else
count_size 0 (n,numerals_size)
exception Negative
let numerals = ['I';'V';'X';'L';'C';'D';'M']
let roman (n:int) : string =
let size = roman_size n in
let x = String.make size 'M' in
let ( ++ ) c k = String.set x k c; k-1 in
let digit d one five ten k =
match d with
| 0 -> k
| 1 -> one ++ k
| 2 -> one ++ (one ++ k)
| 3 -> one ++ (one ++ (one ++ k))
| 4 -> one ++ (five ++ k)
| 5 -> five ++ k
| 6 -> five ++ (one ++ k)
| 7 -> five ++ (one ++ (one ++ k))
| 8 -> five ++ (one ++ (one ++ (one ++ k)))
| 9 -> one ++ (ten ++ k)
| _ -> raise Negative
in
let rec count k = function
| 0,_ -> ()
| n,[_] -> ()
| n,(one :: five :: ((ten :: _) as next)) ->
count (digit (n mod 10) one five ten k) (n/10, next)
| _ -> assert false
in
count (size-1) (n,numerals); x
let debug = false
let () = if debug then begin
let rec upto i j = if i > j then [] else i :: upto (i+1) j in
let test n = arabic (roman n) = n in
let _ = List.for_all ~f:test (upto 0 5000) in
()
end
end
open Roman
type t = string
exception Bad of string
let raise_bad msg = raise (Bad msg)
let of_string_exn s' =
let s = String.uppercase s' in
try ignore (arabic s); s
with BadNumeral c -> raise_bad (sprintf "roman numeral %s contains illegal character %c" s' c)
let of_string s =
try Some (of_string_exn s)
with Bad _ -> None
let of_int_exn i =
if i > 0 then
roman i
else
raise_bad (sprintf "cannot convert int %d to roman numeral" i)
let of_int i =
try Some (of_int_exn i)
with Bad _ -> None
let to_string t = t
let to_int t = arabic t