-
Notifications
You must be signed in to change notification settings - Fork 29
/
common.fs
182 lines (152 loc) · 5.31 KB
/
common.fs
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
178
179
180
181
182
module FVim.common
open System.Threading.Tasks
open System
open System.Diagnostics
open System.Buffers
open Avalonia.Media
let mkparams1 (t1: 'T1) = [| box t1 |]
let mkparams2 (t1: 'T1) (t2: 'T2) = [| box t1; box t2 |]
let mkparams3 (t1: 'T1) (t2: 'T2) (t3: 'T3) = [| box t1; box t2; box t3 |]
let mkparams4 (t1: 'T1) (t2: 'T2) (t3: 'T3) (t4: 'T4) = [| box t1; box t2; box t3; box t4|]
let mkparams5 (t1: 'T1) (t2: 'T2) (t3: 'T3) (t4: 'T4) (t5: 'T5) = [| box t1; box t2; box t3; box t4; box t5|]
let ParseUInt16 (x: string) =
match System.UInt16.TryParse x with
| true, x -> Some x
| _ -> None
let (|ParseUInt16|_|) (x: string) =
match System.UInt16.TryParse x with
| true, x -> Some x
| _ -> None
let (|ParseInt32|_|) (x: string) =
match System.Int32.TryParse x with
| true, x -> Some x
| _ -> None
let (|ParseIp|_|) (x: string) =
match System.Net.IPAddress.TryParse x with
| true, x -> Some x
| _ ->
try
System.Net.Dns.GetHostEntry(x).AddressList
|> Seq.tryFind (fun addr -> addr.AddressFamily = System.Net.Sockets.AddressFamily.InterNetwork)
with _ -> None
let (|ObjArray|_|) (x:obj) =
match x with
| :? (obj[]) as x -> Some x
| :? (obj list) as x -> Some(Array.ofList x)
| :? (obj seq) as x -> Some(Array.ofSeq x)
| _ -> None
let (|Bool|_|) (x:obj) =
match x with
| :? bool as x -> Some x
| _ -> None
let (|String|_|) (x:obj) =
match x with
| :? string as x -> Some x
| _ -> None
let IsString (x:obj) =
match x with
| :? string as x -> Some x
| _ -> None
let Integer32 (x:obj) =
match x with
| :? int32 as x -> Some(int32 x)
| :? int16 as x -> Some(int32 x)
| :? int8 as x -> Some(int32 x)
| :? uint16 as x -> Some(int32 x)
| :? uint32 as x -> Some(int32 x)
| :? uint8 as x -> Some(int32 x)
| _ -> None
let (|Integer32|_|) (x:obj) =
match x with
| :? int32 as x -> Some(int32 x)
| :? int16 as x -> Some(int32 x)
| :? int8 as x -> Some(int32 x)
| :? uint16 as x -> Some(int32 x)
| :? uint32 as x -> Some(int32 x)
| :? uint8 as x -> Some(int32 x)
| _ -> None
let (|Float|_|) (x:obj) =
match x with
| Integer32 x -> Some(float x)
| :? single as x -> Some(float x)
| :? float as x -> Some(float x)
| _ -> None
let (|ByteArray|_|) (x:obj) =
match x with
| :? (byte[]) as x -> Some(x)
| _ -> None
// converts to bool in a desperate (read: JavaScript) attempt
let (|ForceBool|_|) (x:obj) =
match x with
| Bool x -> Some x
| String("v:true")
| String("true") -> Some true
| String("v:false")
| String("false") -> Some false
| String(ParseInt32 x) when x <> 0 -> Some true
| String(ParseInt32 x) when x = 0 -> Some false
| String("") -> Some false
| String(_) -> Some true
| _ -> None
type hashmap<'a, 'b> = System.Collections.Generic.Dictionary<'a, 'b>
let hashmap (xs: seq<'a*'b>) = new hashmap<'a,'b>(xs |> Seq.map (fun (a,b) -> System.Collections.Generic.KeyValuePair(a,b)))
let escapeArgs: string seq -> string seq = Seq.map (fun (x: string) -> if x.Contains(' ') then sprintf "\"%s\"" (x.Replace("\"", "\\\"")) else x)
let join (xs: string seq) = System.String.Join(" ", xs)
let inline (>>=) (x: 'a option) (f: 'a -> 'b option) =
match x with
| Some x -> f x
| _ -> None
let inline (>?=) (x: Result<'a, 'e>) (f: 'a -> Result<'b, 'e>) =
match x with
| Ok result -> f result
| Error err -> Error err
let run (t: Task) =
Task.Run(fun () -> t) |> ignore
let runSync (t: Task) =
t.ConfigureAwait(false).GetAwaiter().GetResult()
let rec read (stream: System.IO.Stream) (buf: Memory<byte>) =
task {
let! n = stream.ReadAsync buf
if n = 0 then failwith "read"
if n < buf.Length then return! read stream (buf.Slice n)
}
let write (stream: System.IO.Stream) (buf: ReadOnlyMemory<byte>) =
stream.WriteAsync(buf)
let inline toInt32LE (x: _[]) =
int32 (x.[0])
||| (int32(x.[1]) <<< 8)
||| (int32(x.[2]) <<< 16)
||| (int32(x.[3]) <<< 24)
let inline fromInt32LE (x: int32) =
[|
byte (x)
byte (x >>> 8)
byte (x >>> 16)
byte (x >>> 24)
|]
let newProcess prog args stderrenc =
let args = args |> escapeArgs |> join
let psi = ProcessStartInfo(prog, args)
psi.CreateNoWindow <- true
psi.ErrorDialog <- false
psi.RedirectStandardError <- true
psi.RedirectStandardInput <- true
psi.RedirectStandardOutput <- true
psi.StandardErrorEncoding <- stderrenc
psi.UseShellExecute <- false
psi.WindowStyle <- ProcessWindowStyle.Hidden
psi.WorkingDirectory <- Environment.CurrentDirectory
let p = new Process()
p.StartInfo <- psi
p.EnableRaisingEvents <- true
// in case the parent crashed and we happen to be running Windows(tm)...
// TODO need further investigation.
ignore <| AppDomain.CurrentDomain.ProcessExit.Subscribe(fun _ -> try p.Kill(true) with _ -> ())
p
let _d x = Option.defaultValue x
let notNull x = isNull x |> not
let inline (?->) x f =
if isNull x then ValueNone
else ValueSome(f x)
let removeAlpha (x:Color) =
Color(255uy, x.R, x.G, x.B)