[2] | 1 | module String_map = Map.Make (String)
|
---|
| 2 | type t = {
|
---|
[3] | 3 | id: Id.t;
|
---|
| 4 | title: string;
|
---|
| 5 | authors: Person.Set.t;
|
---|
| 6 | date: Date.t;
|
---|
| 7 | string_map: string String_map.t;
|
---|
| 8 | stringset_map: String_set.t String_map.t;
|
---|
| 9 | body: string;
|
---|
| 10 | }
|
---|
[2] | 11 |
|
---|
[3] | 12 | let blank ?(id=(Id.generate ())) () = {
|
---|
| 13 | id;
|
---|
| 14 | title = "";
|
---|
| 15 | authors = Person.Set.empty;
|
---|
| 16 | date = Date.({ created = now (); edited = ""});
|
---|
| 17 | string_map = String_map.empty;
|
---|
| 18 | stringset_map = String_map.empty;
|
---|
| 19 | body = "";
|
---|
| 20 | }
|
---|
[2] | 21 |
|
---|
| 22 | let compare = Stdlib.compare
|
---|
| 23 | let newest a b = Date.(compare a.date b.date)
|
---|
| 24 | let oldest a b = Date.(compare b.date a.date)
|
---|
| 25 | let str key m = try String_map.find (String.lowercase_ascii key) m.string_map with Not_found -> ""
|
---|
| 26 | let set key m = try String_map.find (String.lowercase_ascii key) m.stringset_map with Not_found -> String_set.empty
|
---|
| 27 | let str_set key m = String_set.to_string @@ set key m
|
---|
| 28 | let with_str_set m key str = { m with stringset_map = String_map.add (String.lowercase_ascii key) (String_set.of_string str) m.stringset_map }
|
---|
| 29 |
|
---|
| 30 | let with_kv x (k,v) =
|
---|
[3] | 31 | let trim = String.trim in
|
---|
| 32 | match String.lowercase_ascii k with
|
---|
| 33 | | "body" -> { x with body = String.trim v }
|
---|
| 34 | | "title"-> { x with title = trim v }
|
---|
| 35 | | "id" -> (match v with "" -> x | s -> { x with id = s })
|
---|
| 36 | | "author"
|
---|
| 37 | | "authors" -> { x with authors = Person.Set.of_string (trim v)}
|
---|
| 38 | | "date" -> { x with date = Date.{ x.date with created = Date.of_string v }}
|
---|
| 39 | | "date-edited"-> { x with date = Date.{ x.date with edited = Date.of_string v }}
|
---|
| 40 | | "licences" | "topics" | "keywords" | "series" as k -> with_str_set x k v
|
---|
| 41 | | k -> { x with string_map = String_map.add k (trim v) x.string_map }
|
---|
[2] | 42 |
|
---|
[3] | 43 | let kv_of_string line = match Str.(bounded_split (regexp ": *")) line 2 with
|
---|
| 44 | | [ key; value ] -> Str.(replace_first (regexp "^#\\+") "" key), value
|
---|
| 45 | | [ key ] -> Str.(replace_first (regexp "^#\\+") "" key), ""
|
---|
| 46 | | _ -> "",""
|
---|
[2] | 47 |
|
---|
| 48 | let of_header front_matter =
|
---|
[3] | 49 | let fields = List.map kv_of_string (Str.(split (regexp "\n")) front_matter) in
|
---|
| 50 | List.fold_left with_kv (blank ~id:Id.nil ()) fields
|
---|
[2] | 51 |
|
---|
| 52 | let front_matter_body_split s =
|
---|
[3] | 53 | if Str.(string_match (regexp ".*:.*")) s 0
|
---|
| 54 | then match Str.(bounded_split (regexp "^$")) s 2 with
|
---|
| 55 | | front::body::[] -> (front, body)
|
---|
| 56 | | _ -> ("", s)
|
---|
| 57 | else ("", s)
|
---|
[2] | 58 |
|
---|
| 59 | let of_string s =
|
---|
[3] | 60 | let front_matter, body = front_matter_body_split s in
|
---|
| 61 | try
|
---|
| 62 | let note = { (of_header front_matter) with body } in
|
---|
| 63 | if note.id <> Id.nil then Ok note else Error "Missing ID header"
|
---|
| 64 | with _ -> Error ("Failed parsing" ^ s)
|
---|
[2] | 65 |
|
---|
| 66 | let to_string x =
|
---|
[3] | 67 | let has_len v = String.length v > 0 in
|
---|
| 68 | let s field value = if has_len value then field ^ ": " ^ value ^ "\n" else "" in
|
---|
| 69 | let a value = if Person.Set.is_empty value then "" else "Authors: " ^ Person.Set.to_string value ^ "\n" in
|
---|
| 70 | let d field value = match value with "" -> "" | s -> field ^ ": " ^ Date.rfc_string s ^ "\n" in
|
---|
| 71 | let rows = [
|
---|
| 72 | s "ID" x.id;
|
---|
| 73 | d "Date" x.date.Date.created;
|
---|
| 74 | d "Edited" x.date.Date.edited;
|
---|
| 75 | s "Title" x.title;
|
---|
| 76 | a x.authors;
|
---|
| 77 | s "Licences" (str_set "licences" x);
|
---|
| 78 | s "Topics" (str_set "topics" x);
|
---|
| 79 | s "Keywords" (str_set "keywords" x);
|
---|
| 80 | s "Series" (str_set "series" x);
|
---|
| 81 | s "Abstract" (str "abstract" x);
|
---|
| 82 | s "Alias" (str "Alias" x)
|
---|
| 83 | ] in
|
---|
| 84 | String.concat "" rows ^ "\n" ^ x.body
|
---|
[2] | 85 |
|
---|
| 86 | let string_alias t =
|
---|
[3] | 87 | let is_reserved = function
|
---|
| 88 | | '!' | '*' | '\'' | '(' | ')' | ';' | ':' | '@' | '&' | '=' | '+' | '$'
|
---|
| 89 | | ',' | '/' | '?' | '#' | '[' | ']' | ' ' | '\t' | '\x00' -> true
|
---|
| 90 | | _ -> false
|
---|
| 91 | in
|
---|
| 92 | let b = Buffer.create (String.length t) in
|
---|
| 93 | let filter char =
|
---|
| 94 | let open Buffer in
|
---|
| 95 | if is_reserved char then (try (if nth b (pred (length b)) <> '-' then add_char b '-') with Invalid_argument _ -> prerr_endline "reserved")
|
---|
| 96 | else add_char b char
|
---|
| 97 | in
|
---|
| 98 | String.(iter filter (lowercase_ascii t));
|
---|
| 99 | Buffer.contents b
|
---|
[2] | 100 |
|
---|
| 101 | let alias t = match str "alias" t with "" -> string_alias t.title | x -> x
|
---|
[3] | 102 | let short_id t = Id.short t.id
|
---|