diff --git a/index.js b/index.js index 386c9d00..8ca251f6 100644 --- a/index.js +++ b/index.js @@ -1137,26 +1137,27 @@ function _Debug_regionToString(region) { "on lines " + region.start.line + " through " + region.end.line ); } -var $gren_lang$core$Dict$foldl = F3(function(func, acc, dict) { - foldl: - while (true) { - if (dict.$ === 'RBEmpty_gren_builtin') { - return acc; - } else { - var key = dict.b; - var value = dict.c; - var left = dict.d; - var right = dict.e; - var $temp$func = func, - $temp$acc = A3(func, key, value, A3($gren_lang$core$Dict$foldl, func, acc, left)), - $temp$dict = right; - func = $temp$func; - acc = $temp$acc; - dict = $temp$dict; - continue foldl; - } +var $gren_lang$core$Dict$foldl$ = function(func, acc, dict) { + foldl: + while (true) { + if (dict.$ === 'RBEmpty_gren_builtin') { + return acc; + } else { + var key = dict.b; + var value = dict.c; + var left = dict.d; + var right = dict.e; + var $temp$func = func, + $temp$acc = A3(func, key, value, $gren_lang$core$Dict$foldl$(func, acc, left)), + $temp$dict = right; + func = $temp$func; + acc = $temp$acc; + dict = $temp$dict; + continue foldl; } - }); + } +}; +var $gren_lang$core$Dict$foldl = F3($gren_lang$core$Dict$foldl$); var _Array_length = function (array) { @@ -1502,7 +1503,7 @@ var $gren_lang$core$Maybe$Just = function (a) { var $gren_lang$core$Maybe$Nothing = { $: 'Nothing' }; var $gren_lang$core$Array$pushLast = _Array_push; var $gren_lang$core$Dict$keys = function(dict) { - return A3($gren_lang$core$Dict$foldl, F3(function(key, value, keyArray) { + return $gren_lang$core$Dict$foldl$(F3(function(key, value, keyArray) { return A2($gren_lang$core$Array$pushLast, key, keyArray); }), [ ], dict); }; @@ -2275,88 +2276,90 @@ var $gren_lang$core$Char$isAlphaNum = function(_char) { }; var $gren_lang$core$Array$length = _Array_length; var $gren_lang$core$String$uncons = _String_uncons; -var $gren_lang$core$Json$Decode$errorOneOf = F2(function(i, error) { - return _Utils_ap('\n\n(', _Utils_ap($gren_lang$core$String$fromInt(i + 1), _Utils_ap(') ', $gren_lang$core$Json$Decode$indent($gren_lang$core$Json$Decode$errorToString(error))))); - }); +var $gren_lang$core$Json$Decode$errorOneOf$ = function(i, error) { + return _Utils_ap('\n\n(', _Utils_ap($gren_lang$core$String$fromInt(i + 1), _Utils_ap(') ', $gren_lang$core$Json$Decode$indent($gren_lang$core$Json$Decode$errorToString(error))))); +}; +var $gren_lang$core$Json$Decode$errorOneOf = F2($gren_lang$core$Json$Decode$errorOneOf$); var $gren_lang$core$Json$Decode$errorToString = function(error) { return A2($gren_lang$core$Json$Decode$errorToStringHelp, error, [ ]); }; -var $gren_lang$core$Json$Decode$errorToStringHelp = F2(function(error, context) { - errorToStringHelp: - while (true) { - switch (error.$) { - case 'Field': - var f = error.a; - var err = error.b; - var isSimple = function () { - var _v1 = $gren_lang$core$String$uncons(f); - if (_v1.$ === 'Nothing') { - return false; - } else { - var _v2 = _v1.a; - var _char = _v2.first; - var rest = _v2.rest; - return $gren_lang$core$Char$isAlpha(_char) && A2($gren_lang$core$String$all, $gren_lang$core$Char$isAlphaNum, rest); - } - }(); - var fieldName = isSimple ? _Utils_ap('.', f) : _Utils_ap('[\'', _Utils_ap(f, '\']')); - var $temp$error = err, - $temp$context = _Utils_ap([ fieldName ], context); - error = $temp$error; - context = $temp$context; - continue errorToStringHelp; - case 'Index': - var i = error.a; - var err = error.b; - var indexName = _Utils_ap('[', _Utils_ap($gren_lang$core$String$fromInt(i), ']')); - var $temp$error = err, - $temp$context = _Utils_ap([ indexName ], context); - error = $temp$error; - context = $temp$context; - continue errorToStringHelp; - case 'OneOf': - var errors = error.a; - switch (errors.length) { - case 0: - return _Utils_ap('Ran into a Json.Decode.oneOf with no possibilities', function () { - if (context.length === 0) { - return '!'; - } else { - return _Utils_ap(' at json', A2($gren_lang$core$String$join, '', context)); - } - }()); - case 1: - var err = errors[0]; - var $temp$error = err, - $temp$context = context; - error = $temp$error; - context = $temp$context; - continue errorToStringHelp; - default: - var starter = function () { +var $gren_lang$core$Json$Decode$errorToStringHelp$ = function(error, context) { + errorToStringHelp: + while (true) { + switch (error.$) { + case 'Field': + var f = error.a; + var err = error.b; + var isSimple = function () { + var _v1 = $gren_lang$core$String$uncons(f); + if (_v1.$ === 'Nothing') { + return false; + } else { + var _v2 = _v1.a; + var _char = _v2.first; + var rest = _v2.rest; + return $gren_lang$core$Char$isAlpha(_char) && A2($gren_lang$core$String$all, $gren_lang$core$Char$isAlphaNum, rest); + } + }(); + var fieldName = isSimple ? _Utils_ap('.', f) : _Utils_ap('[\'', _Utils_ap(f, '\']')); + var $temp$error = err, + $temp$context = _Utils_ap([ fieldName ], context); + error = $temp$error; + context = $temp$context; + continue errorToStringHelp; + case 'Index': + var i = error.a; + var err = error.b; + var indexName = _Utils_ap('[', _Utils_ap($gren_lang$core$String$fromInt(i), ']')); + var $temp$error = err, + $temp$context = _Utils_ap([ indexName ], context); + error = $temp$error; + context = $temp$context; + continue errorToStringHelp; + case 'OneOf': + var errors = error.a; + switch (errors.length) { + case 0: + return _Utils_ap('Ran into a Json.Decode.oneOf with no possibilities', function () { if (context.length === 0) { - return 'Json.Decode.oneOf'; + return '!'; } else { - return _Utils_ap('The Json.Decode.oneOf at json', A2($gren_lang$core$String$join, '', context)); + return _Utils_ap(' at json', A2($gren_lang$core$String$join, '', context)); } - }(); - var introduction = _Utils_ap(starter, _Utils_ap(' failed in the following ', _Utils_ap($gren_lang$core$String$fromInt($gren_lang$core$Array$length(errors)), ' ways:'))); - return A2($gren_lang$core$String$join, '\n\n', _Utils_ap([ introduction ], A2($gren_lang$core$Array$indexedMap, $gren_lang$core$Json$Decode$errorOneOf, errors))); + }()); + case 1: + var err = errors[0]; + var $temp$error = err, + $temp$context = context; + error = $temp$error; + context = $temp$context; + continue errorToStringHelp; + default: + var starter = function () { + if (context.length === 0) { + return 'Json.Decode.oneOf'; + } else { + return _Utils_ap('The Json.Decode.oneOf at json', A2($gren_lang$core$String$join, '', context)); + } + }(); + var introduction = _Utils_ap(starter, _Utils_ap(' failed in the following ', _Utils_ap($gren_lang$core$String$fromInt($gren_lang$core$Array$length(errors)), ' ways:'))); + return A2($gren_lang$core$String$join, '\n\n', _Utils_ap([ introduction ], A2($gren_lang$core$Array$indexedMap, $gren_lang$core$Json$Decode$errorOneOf, errors))); + } + default: + var msg = error.a; + var json = error.b; + var introduction = function () { + if (context.length === 0) { + return 'Problem with the given value:\n\n'; + } else { + return _Utils_ap('Problem with the value at json', _Utils_ap(A2($gren_lang$core$String$join, '', context), ':\n\n ')); } - default: - var msg = error.a; - var json = error.b; - var introduction = function () { - if (context.length === 0) { - return 'Problem with the given value:\n\n'; - } else { - return _Utils_ap('Problem with the value at json', _Utils_ap(A2($gren_lang$core$String$join, '', context), ':\n\n ')); - } - }(); - return _Utils_ap(introduction, _Utils_ap($gren_lang$core$Json$Decode$indent(A2($gren_lang$core$Json$Encode$encode, 4, json)), _Utils_ap('\n\n', msg))); - } + }(); + return _Utils_ap(introduction, _Utils_ap($gren_lang$core$Json$Decode$indent(A2($gren_lang$core$Json$Encode$encode, 4, json)), _Utils_ap('\n\n', msg))); } - }); + } +}; +var $gren_lang$core$Json$Decode$errorToStringHelp = F2($gren_lang$core$Json$Decode$errorToStringHelp$); var $gren_lang$core$Basics$True = { $: 'True' }; var $gren_lang$core$Result$isOk = function(result) { if (result.$ === 'Ok') { @@ -2423,28 +2426,29 @@ var $gren_lang$core$Dict$balance$ = function(color, key, value, left, right) { }; var $gren_lang$core$Dict$balance = F5($gren_lang$core$Dict$balance$); var $gren_lang$core$Basics$compare = _Utils_compare; -var $gren_lang$core$Dict$setHelp = F3(function(key, value, dict) { - if (dict.$ === 'RBEmpty_gren_builtin') { - return $gren_lang$core$Dict$RBNode_gren_builtin$($gren_lang$core$Dict$Red, key, value, $gren_lang$core$Dict$RBEmpty_gren_builtin, $gren_lang$core$Dict$RBEmpty_gren_builtin); - } else { - var nColor = dict.a; - var nKey = dict.b; - var nValue = dict.c; - var nLeft = dict.d; - var nRight = dict.e; - var _v1 = A2($gren_lang$core$Basics$compare, key, nKey); - switch (_v1.$) { - case 'LT': - return $gren_lang$core$Dict$balance$(nColor, nKey, nValue, A3($gren_lang$core$Dict$setHelp, key, value, nLeft), nRight); - case 'EQ': - return $gren_lang$core$Dict$RBNode_gren_builtin$(nColor, nKey, value, nLeft, nRight); - default: - return $gren_lang$core$Dict$balance$(nColor, nKey, nValue, nLeft, A3($gren_lang$core$Dict$setHelp, key, value, nRight)); - } +var $gren_lang$core$Dict$setHelp$ = function(key, value, dict) { + if (dict.$ === 'RBEmpty_gren_builtin') { + return $gren_lang$core$Dict$RBNode_gren_builtin$($gren_lang$core$Dict$Red, key, value, $gren_lang$core$Dict$RBEmpty_gren_builtin, $gren_lang$core$Dict$RBEmpty_gren_builtin); + } else { + var nColor = dict.a; + var nKey = dict.b; + var nValue = dict.c; + var nLeft = dict.d; + var nRight = dict.e; + var _v1 = A2($gren_lang$core$Basics$compare, key, nKey); + switch (_v1.$) { + case 'LT': + return $gren_lang$core$Dict$balance$(nColor, nKey, nValue, $gren_lang$core$Dict$setHelp$(key, value, nLeft), nRight); + case 'EQ': + return $gren_lang$core$Dict$RBNode_gren_builtin$(nColor, nKey, value, nLeft, nRight); + default: + return $gren_lang$core$Dict$balance$(nColor, nKey, nValue, nLeft, $gren_lang$core$Dict$setHelp$(key, value, nRight)); } - }); + } +}; +var $gren_lang$core$Dict$setHelp = F3($gren_lang$core$Dict$setHelp$); var $gren_lang$core$Dict$set$ = function(key, value, dict) { - var _v0 = A3($gren_lang$core$Dict$setHelp, key, value, dict); + var _v0 = $gren_lang$core$Dict$setHelp$(key, value, dict); if ((_v0.$ === 'RBNode_gren_builtin') && (_v0.a.$ === 'Red')) { var _v1 = _v0.a; var k = _v0.b; @@ -2664,13 +2668,15 @@ var $gren_lang$node$Node$defineProgram = function(config) { var $author$project$Main$ExistanceChecked = function (a) { return { $: 'ExistanceChecked', a: a }; }; -var $gren_lang$core$Basics$composeL$ = function(g, f, x) { - return g(f(x)); +var $gren_lang$core$Basics$composeL$ = function(g, f) { + return function(x) { + return g(f(x)); + }; }; -var $gren_lang$core$Basics$composeL = F3($gren_lang$core$Basics$composeL$); +var $gren_lang$core$Basics$composeL = F2($gren_lang$core$Basics$composeL$); var $gren_lang$core$Task$onError = _Scheduler_onError; var $gren_lang$core$Task$attempt$ = function(resultToMessage, task) { - return $gren_lang$core$Task$command($gren_lang$core$Task$Perform(A2($gren_lang$core$Task$onError, A2($gren_lang$core$Basics$composeL, A2($gren_lang$core$Basics$composeL, $gren_lang$core$Task$succeed, resultToMessage), $gren_lang$core$Result$Err), A2($gren_lang$core$Task$andThen, A2($gren_lang$core$Basics$composeL, A2($gren_lang$core$Basics$composeL, $gren_lang$core$Task$succeed, resultToMessage), $gren_lang$core$Result$Ok), task)))); + return $gren_lang$core$Task$command($gren_lang$core$Task$Perform(A2($gren_lang$core$Task$onError, $gren_lang$core$Basics$composeL$($gren_lang$core$Basics$composeL$($gren_lang$core$Task$succeed, resultToMessage), $gren_lang$core$Result$Err), A2($gren_lang$core$Task$andThen, $gren_lang$core$Basics$composeL$($gren_lang$core$Basics$composeL$($gren_lang$core$Task$succeed, resultToMessage), $gren_lang$core$Result$Ok), task)))); }; var $gren_lang$core$Task$attempt = F2($gren_lang$core$Task$attempt$); var $gren_lang$core$Basics$identity = function(x) { @@ -2685,11 +2691,11 @@ var $gren_lang$node$Init$unwrap = function(_v0) { }; var $gren_lang$node$Init$await$ = function(_v0, fn) { var task = _v0.a; - return $gren_lang$node$Internal$Init$Task(A2($gren_lang$core$Task$andThen, A2($gren_lang$core$Basics$composeL, $gren_lang$node$Init$unwrap, fn), task)); + return $gren_lang$node$Internal$Init$Task(A2($gren_lang$core$Task$andThen, $gren_lang$core$Basics$composeL$($gren_lang$node$Init$unwrap, fn), task)); }; var $gren_lang$node$Init$await = F2($gren_lang$node$Init$await$); var $gren_lang$node$Init$awaitTask$ = function(task, fn) { - return $gren_lang$node$Internal$Init$Task(A2($gren_lang$core$Task$andThen, A2($gren_lang$core$Basics$composeL, $gren_lang$node$Init$unwrap, fn), task)); + return $gren_lang$node$Internal$Init$Task(A2($gren_lang$core$Task$andThen, $gren_lang$core$Basics$composeL$($gren_lang$node$Init$unwrap, fn), task)); }; var $gren_lang$node$Init$awaitTask = F2($gren_lang$node$Init$awaitTask$); @@ -3401,21 +3407,6 @@ var $gren_lang$node$FileSystem$checkAccess$ = function(_v0, permissions, path) { return A2(_FileSystem_access, permissions, path); }; var $gren_lang$node$FileSystem$checkAccess = F3($gren_lang$node$FileSystem$checkAccess$); -var $gren_lang$core$Basics$sub = _Basics_sub; -var $author$project$Main$countDown = F2(function(n, rec) { - countDown: - while (true) { - if (_Utils_cmp(n, 0) < 1) { - return 0; - } else { - var $temp$n = n - 1, - $temp$rec = rec; - n = $temp$n; - rec = $temp$rec; - continue countDown; - } - } - }); var $gren_lang$core$Array$slice = _Array_slice; var $gren_lang$core$Array$dropFirst$ = function(n, array) { return A3($gren_lang$core$Array$slice, n, $gren_lang$core$Array$length(array), array); @@ -3430,36 +3421,37 @@ var $gren_lang$core$Task$execute = function(task) { }; var $gren_lang$node$FileSystem$Path$fromPosixString = _FilePath_fromPosix; var $gren_lang$node$FileSystem$Path$fromWin32String = _FilePath_fromWin32; -var $gren_lang$core$Dict$get = F2(function(targetKey, dict) { - get: - while (true) { - if (dict.$ === 'RBEmpty_gren_builtin') { - return $gren_lang$core$Maybe$Nothing; - } else { - var key = dict.b; - var value = dict.c; - var left = dict.d; - var right = dict.e; - var _v1 = A2($gren_lang$core$Basics$compare, targetKey, key); - switch (_v1.$) { - case 'LT': - var $temp$targetKey = targetKey, - $temp$dict = left; - targetKey = $temp$targetKey; - dict = $temp$dict; - continue get; - case 'EQ': - return $gren_lang$core$Maybe$Just(value); - default: - var $temp$targetKey = targetKey, - $temp$dict = right; - targetKey = $temp$targetKey; - dict = $temp$dict; - continue get; - } +var $gren_lang$core$Dict$get$ = function(targetKey, dict) { + get: + while (true) { + if (dict.$ === 'RBEmpty_gren_builtin') { + return $gren_lang$core$Maybe$Nothing; + } else { + var key = dict.b; + var value = dict.c; + var left = dict.d; + var right = dict.e; + var _v1 = A2($gren_lang$core$Basics$compare, targetKey, key); + switch (_v1.$) { + case 'LT': + var $temp$targetKey = targetKey, + $temp$dict = left; + targetKey = $temp$targetKey; + dict = $temp$dict; + continue get; + case 'EQ': + return $gren_lang$core$Maybe$Just(value); + default: + var $temp$targetKey = targetKey, + $temp$dict = right; + targetKey = $temp$targetKey; + dict = $temp$dict; + continue get; } } - }); + } +}; +var $gren_lang$core$Dict$get = F2($gren_lang$core$Dict$get$); var $gren_lang$node$Node$getEnvironmentVariables = _Node_getEnvironmentVariables; var $gren_lang$node$FileSystem$homeDirectory = function(_v0) { return _FileSystem_homeDir; @@ -3569,11 +3561,11 @@ var $author$project$Main$makeLocalPath$ = function(platform, homeDir, envVars) { var startPath = function () { switch (platform.$) { case 'Win32': - return $gren_lang$core$Maybe$withDefault$($gren_lang$node$FileSystem$Path$prepend$(homeDir, $gren_lang$node$FileSystem$Path$fromPosixString('AppData/Local')), $gren_lang$core$Maybe$map$($gren_lang$node$FileSystem$Path$fromWin32String, A2($gren_lang$core$Dict$get, 'LOCALAPPDATA', envVars))); + return $gren_lang$core$Maybe$withDefault$($gren_lang$node$FileSystem$Path$prepend$(homeDir, $gren_lang$node$FileSystem$Path$fromPosixString('AppData/Local')), $gren_lang$core$Maybe$map$($gren_lang$node$FileSystem$Path$fromWin32String, $gren_lang$core$Dict$get$('LOCALAPPDATA', envVars))); case 'Darwin': return $gren_lang$node$FileSystem$Path$prepend$(homeDir, $gren_lang$node$FileSystem$Path$fromPosixString('Library/Caches')); default: - return $gren_lang$core$Maybe$withDefault$($gren_lang$node$FileSystem$Path$append$($gren_lang$node$FileSystem$Path$fromPosixString('.cache'), homeDir), $gren_lang$core$Maybe$map$($gren_lang$node$FileSystem$Path$fromPosixString, A2($gren_lang$core$Dict$get, 'XDG_CACHE_HOME', envVars))); + return $gren_lang$core$Maybe$withDefault$($gren_lang$node$FileSystem$Path$append$($gren_lang$node$FileSystem$Path$fromPosixString('.cache'), homeDir), $gren_lang$core$Maybe$map$($gren_lang$node$FileSystem$Path$fromPosixString, $gren_lang$core$Dict$get$('XDG_CACHE_HOME', envVars))); } }(); var filename = function () { @@ -3789,54 +3781,56 @@ var $gren_lang$core$Bytes$Encode$getLength = function(builder) { }; var $gren_lang$core$Bytes$LE = { $: 'LE' }; var $gren_lang$core$Array$foldl = _Array_foldl; -var $gren_lang$core$Bytes$Encode$write = F3(function(builder, mb, offset) { - switch (builder.$) { - case 'I8': - var n = builder.a; - return A3(_Bytes_write_i8, mb, offset, n); - case 'I16': - var e = builder.a; - var n = builder.b; - return A4(_Bytes_write_i16, mb, offset, n, _Utils_eq(e, $gren_lang$core$Bytes$LE)); - case 'I32': - var e = builder.a; - var n = builder.b; - return A4(_Bytes_write_i32, mb, offset, n, _Utils_eq(e, $gren_lang$core$Bytes$LE)); - case 'U8': - var n = builder.a; - return A3(_Bytes_write_u8, mb, offset, n); - case 'U16': - var e = builder.a; - var n = builder.b; - return A4(_Bytes_write_u16, mb, offset, n, _Utils_eq(e, $gren_lang$core$Bytes$LE)); - case 'U32': - var e = builder.a; - var n = builder.b; - return A4(_Bytes_write_u32, mb, offset, n, _Utils_eq(e, $gren_lang$core$Bytes$LE)); - case 'F32': - var e = builder.a; - var n = builder.b; - return A4(_Bytes_write_f32, mb, offset, n, _Utils_eq(e, $gren_lang$core$Bytes$LE)); - case 'F64': - var e = builder.a; - var n = builder.b; - return A4(_Bytes_write_f64, mb, offset, n, _Utils_eq(e, $gren_lang$core$Bytes$LE)); - case 'Seq': - var bs = builder.b; - return A3($gren_lang$core$Bytes$Encode$writeSequence, bs, mb, offset); - case 'Utf8': - var s = builder.b; - return A3(_Bytes_write_string, mb, offset, s); - default: - var bs = builder.a; - return A3(_Bytes_write_bytes, mb, offset, bs); - } - }); -var $gren_lang$core$Bytes$Encode$writeSequence = F3(function(builders, mb, offset) { - return A3($gren_lang$core$Array$foldl, F2(function(builder, currentOffset) { - return A3($gren_lang$core$Bytes$Encode$write, builder, mb, currentOffset); - }), offset, builders); - }); +var $gren_lang$core$Bytes$Encode$write$ = function(builder, mb, offset) { + switch (builder.$) { + case 'I8': + var n = builder.a; + return A3(_Bytes_write_i8, mb, offset, n); + case 'I16': + var e = builder.a; + var n = builder.b; + return A4(_Bytes_write_i16, mb, offset, n, _Utils_eq(e, $gren_lang$core$Bytes$LE)); + case 'I32': + var e = builder.a; + var n = builder.b; + return A4(_Bytes_write_i32, mb, offset, n, _Utils_eq(e, $gren_lang$core$Bytes$LE)); + case 'U8': + var n = builder.a; + return A3(_Bytes_write_u8, mb, offset, n); + case 'U16': + var e = builder.a; + var n = builder.b; + return A4(_Bytes_write_u16, mb, offset, n, _Utils_eq(e, $gren_lang$core$Bytes$LE)); + case 'U32': + var e = builder.a; + var n = builder.b; + return A4(_Bytes_write_u32, mb, offset, n, _Utils_eq(e, $gren_lang$core$Bytes$LE)); + case 'F32': + var e = builder.a; + var n = builder.b; + return A4(_Bytes_write_f32, mb, offset, n, _Utils_eq(e, $gren_lang$core$Bytes$LE)); + case 'F64': + var e = builder.a; + var n = builder.b; + return A4(_Bytes_write_f64, mb, offset, n, _Utils_eq(e, $gren_lang$core$Bytes$LE)); + case 'Seq': + var bs = builder.b; + return A3($gren_lang$core$Bytes$Encode$writeSequence, bs, mb, offset); + case 'Utf8': + var s = builder.b; + return A3(_Bytes_write_string, mb, offset, s); + default: + var bs = builder.a; + return A3(_Bytes_write_bytes, mb, offset, bs); + } +}; +var $gren_lang$core$Bytes$Encode$write = F3($gren_lang$core$Bytes$Encode$write$); +var $gren_lang$core$Bytes$Encode$writeSequence$ = function(builders, mb, offset) { + return A3($gren_lang$core$Array$foldl, F2(function(builder, currentOffset) { + return A3($gren_lang$core$Bytes$Encode$write, builder, mb, currentOffset); + }), offset, builders); +}; +var $gren_lang$core$Bytes$Encode$writeSequence = F3($gren_lang$core$Bytes$Encode$writeSequence$); var $gren_lang$core$Bytes$fromString = _Bytes_fromString; @@ -3889,12 +3883,11 @@ var $author$project$Main$init = function(env) { return $gren_lang$node$Init$awaitTask$($gren_lang$node$Node$getEnvironmentVariables, function(envVars) { return $gren_lang$node$Init$awaitTask$($gren_lang$node$FileSystem$homeDirectory(fsPermission), function(homeDir) { var userArgs = $gren_lang$core$Array$dropFirst$(2, env.args); - var useless = A2($author$project$Main$countDown, 10, { }); var useColor = function () { if (terminalConfig.$ === 'Nothing') { return false; } else { - var _v10 = A2($gren_lang$core$Dict$get, 'NO_COLOR', envVars); + var _v10 = $gren_lang$core$Dict$get$('NO_COLOR', envVars); if (_v10.$ === 'Just') { return false; } else { @@ -3903,7 +3896,7 @@ var $author$project$Main$init = function(env) { } }(); var maybePaths = function () { - var _v2 = { arch: env.cpuArchitecture, override: A2($gren_lang$core$Dict$get, 'GREN_BIN', envVars), platform: env.platform }; + var _v2 = { arch: env.cpuArchitecture, override: $gren_lang$core$Dict$get$('GREN_BIN', envVars), platform: env.platform }; _v2$1: while (true) { _v2$5: @@ -4519,6 +4512,7 @@ var $gren_lang$node$FileSystem$makeDirectory$ = function(_v0, options, path) { return A2(_FileSystem_makeDirectory, options, path); }; var $gren_lang$node$FileSystem$makeDirectory = F3($gren_lang$node$FileSystem$makeDirectory$); +var $gren_lang$core$Basics$sub = _Basics_sub; var $gren_lang$core$Array$dropLast$ = function(n, array) { return A3($gren_lang$core$Array$slice, 0, $gren_lang$core$Array$length(array) - n, array); }; @@ -4785,7 +4779,7 @@ var $author$project$Main$update$ = function(msg, model) { var err = msg.a.a; var res = err.a; if (_Utils_eq(res.statusCode, 302)) { - var _v3 = A2($gren_lang$core$Dict$get, 'location', res.headers); + var _v3 = $gren_lang$core$Dict$get$('location', res.headers); if ((_v3.$ === 'Just') && (_v3.a.length === 1)) { var location = _v3.a[0]; return { command: $gren_lang$core$Task$attempt$($author$project$Main$CompilerDownloaded, $author$project$Main$downloadBinary$(model.httpPermission, location)), model: model }; @@ -4828,5 +4822,3 @@ catch (e) { console.error(e); } - -//# sourceMappingURL=data:application/json;base64,{
    "version": 3,
    "sources": [
        "Task",
        "Basics",
        "Dict",
        "Array",
        "Set",
        "String",
        "Json.Encode",
        "Json.Decode",
        "Char",
        "Result",
        "Node",
        "Platform",
        "Platform.Sub",
        "Platform.Cmd",
        "Init",
        "Time",
        "FileSystem",
        "Main",
        "FileSystem.Path",
        "ChildProcess",
        "HttpClient",
        "Terminal",
        "Maybe",
        "Bytes.Encode",
        "Bytes",
        "Stream",
        "HttpServer",
        "Process"
    ],
    "sourcesContent": [
        "effect module Task where { command = MyCmd } exposing\n    ( Task, perform, attempt, execute\n    , andThen, await, succeed, fail, sequence\n    , map, map2, map3, map4, map5\n    , onError, mapError\n    )\n\n{-| Tasks make it easy to describe asynchronous operations that may fail, like\nHTTP requests or writing to a database.\n\n\n@docs Task, perform, attempt, execute\n\n\n## Chains\n\n@docs andThen, await, succeed, fail, sequence\n\n\n## Maps\n\n@docs map, map2, map3, map4, map5\n\n\n## Errors\n\n@docs onError, mapError\n\n-}\n\nimport Array exposing (Array)\nimport Basics exposing ((<<), (|>), Never)\nimport Gren.Kernel.Scheduler\nimport Maybe exposing (Maybe(..))\nimport Platform\nimport Platform.Cmd exposing (Cmd)\nimport Result exposing (Result(..))\n\n\n{-| Here are some common tasks:\n\n  - [`now : Task x Posix`](Time#now)\n  - [`focus : String -> Task Error {}`][focus]\n  - [`sleep : Float -> Task x {}`](Process#sleep)\n\n[focus]: /package/gren-lang/browser/latest/module/Browser.Dom#focus\n\nIn each case we have a `Task` that will resolve successfully with an `a` value\nor unsuccessfully with an `x` value. So `Browser.Dom.focus` we may fail with an\n`Error` if the given ID does not exist. Whereas `Time.now` never fails so\nI cannot be more specific than `x`. No such value will ever exist! Instead it\nalways succeeds with the current POSIX time.\n\nMore generally a task is a _description_ of what you need to do. Like a todo\nlist. Or like a grocery list. Or like GitHub issues. So saying \"the task is\nto tell me the current POSIX time\" does not complete the task! You need\n[`perform`](#perform) tasks or [`attempt`](#attempt) tasks.\n\n-}\ntype alias Task x a =\n    Platform.Task x a\n\n\n\n-- BASICS\n\n\n{-| A task that succeeds immediately when run. It is usually used with\n[`andThen`](#andThen). You can use it like `map` if you want:\n\n    import Time\n\n\n    timeInMillis : Task x Int\n    timeInMillis =\n        Time.now\n            |> andThen (\\t -> succeed (Time.posixToMillis t))\n\n-}\nsucceed : a -> Task x a\nsucceed =\n    Gren.Kernel.Scheduler.succeed\n\n\n{-| A task that fails immediately when run. Like with `succeed`, this can be\nused with `andThen` to check on the outcome of another task.\n\n    type Error\n        = NotFound\n\n    notFound : Task Error a\n    notFound =\n        fail NotFound\n\n-}\nfail : x -> Task x a\nfail =\n    Gren.Kernel.Scheduler.fail\n\n\n\n-- MAPPING\n\n\n{-| Transform a task. Maybe you want to use [`Time`][time] to figure\nout what time it will be in one hour:\n\n    import Task exposing (Task)\n    import Time\n\n\n    timeInOneHour : Task x Time.Posix\n    timeInOneHour =\n        Task.map addAnHour Time.now\n\n    addAnHour : Time.Posix -> Time.Posix\n    addAnHour time =\n        Time.millisToPosix (Time.posixToMillis time + 60 * 60 * 1000)\n\n[time]: Time\n\n-}\nmap : (a -> b) -> Task x a -> Task x b\nmap func taskA =\n    taskA\n        |> andThen (\\a -> succeed (func a))\n\n\n{-| Put the results of two tasks together. For example, if we wanted to know\nthe current month, we could use [`Time`][time] to ask:\n\n    import Task exposing (Task)\n    import Time\n\n\n    getMonth : Task x Int\n    getMonth =\n        Task.map2 Time.toMonth Time.here Time.now\n\n**Note:** Say we were doing HTTP requests instead. `map2` does each task in\norder, so it would try the first request and only continue after it succeeds.\nIf it fails, the whole thing fails!\n\n[time]: Time\n\n-}\nmap2 : (a -> b -> result) -> Task x a -> Task x b -> Task x result\nmap2 func taskA taskB =\n    taskA\n        |> andThen\n            (\\a ->\n                taskB\n                    |> andThen (\\b -> succeed (func a b))\n            )\n\n\n{-| -}\nmap3 : (a -> b -> c -> result) -> Task x a -> Task x b -> Task x c -> Task x result\nmap3 func taskA taskB taskC =\n    taskA\n        |> andThen\n            (\\a ->\n                taskB\n                    |> andThen\n                        (\\b ->\n                            taskC\n                                |> andThen (\\c -> succeed (func a b c))\n                        )\n            )\n\n\n{-| -}\nmap4 : (a -> b -> c -> d -> result) -> Task x a -> Task x b -> Task x c -> Task x d -> Task x result\nmap4 func taskA taskB taskC taskD =\n    taskA\n        |> andThen\n            (\\a ->\n                taskB\n                    |> andThen\n                        (\\b ->\n                            taskC\n                                |> andThen\n                                    (\\c ->\n                                        taskD\n                                            |> andThen (\\d -> succeed (func a b c d))\n                                    )\n                        )\n            )\n\n\n{-| -}\nmap5 : (a -> b -> c -> d -> e -> result) -> Task x a -> Task x b -> Task x c -> Task x d -> Task x e -> Task x result\nmap5 func taskA taskB taskC taskD taskE =\n    taskA\n        |> andThen\n            (\\a ->\n                taskB\n                    |> andThen\n                        (\\b ->\n                            taskC\n                                |> andThen\n                                    (\\c ->\n                                        taskD\n                                            |> andThen\n                                                (\\d ->\n                                                    taskE\n                                                        |> andThen (\\e -> succeed (func a b c d e))\n                                                )\n                                    )\n                        )\n            )\n\n\n{-| Start with a list of tasks, and turn them into a single task that returns a\nlist. The tasks will be run in order one-by-one and if any task fails the whole\nsequence fails.\n\n    sequence [ succeed 1, succeed 2 ] == succeed [ 1, 2 ]\n\n-}\nsequence : Array (Task x a) -> Task x (Array a)\nsequence tasks =\n    Array.foldr (map2 Array.pushFirst) (succeed []) tasks\n\n\n\n-- CHAINING\n\n\n{-| Chain together a task and a callback. The first task will run, and if it is\nsuccessful, you give the result to the callback resulting in another task. This\ntask then gets run. We could use this to make a task that resolves an hour from\nnow:\n\n\n    import Process\n    import Time\n\n    timeInOneHour : Task x Time.Posix\n    timeInOneHour =\n        Process.sleep (60 * 60 * 1000)\n            |> andThen (\\_ -> Time.now)\n\nFirst the process sleeps for an hour **and then** it tells us what time it is.\n\n-}\nandThen : (a -> Task x b) -> Task x a -> Task x b\nandThen =\n    Gren.Kernel.Scheduler.andThen\n\n\n{-| This is like [andThen](andThen) but the arguments are reversed. The callback\nis the last argument, instead of the first. This makes it easier to write imperative\ncode where each callback involves more logic.\n\n    import Process\n    import Time\n\n    timeInOneHour : Task x Time.Posix\n    timeInOneHour =\n        Task.await (Process.sleep <| 60 * 60 * 1000) <| \\_ ->\n            Time.now\n\n(a)wait for an hour, then fetch the current time.\n\n-}\nawait : Task x a -> (a -> Task x b) -> Task x b\nawait tsk callback =\n    Gren.Kernel.Scheduler.andThen callback tsk\n\n\n-- ERRORS\n\n\n{-| Recover from a failure in a task. If the given task fails, we use the\ncallback to recover.\n\n    fail \"file not found\"\n      |> onError (\\msg -> succeed 42)\n      -- succeed 42\n\n    succeed 9\n      |> onError (\\msg -> succeed 42)\n      -- succeed 9\n\n-}\nonError : (x -> Task y a) -> Task x a -> Task y a\nonError =\n    Gren.Kernel.Scheduler.onError\n\n\n{-| Transform the error value. This can be useful if you need a bunch of error\ntypes to match up.\n\n    type Error\n        = Http Http.Error\n        | WebGL WebGL.Error\n\n    getResources : Task Error Resource\n    getResources =\n        sequence\n            [ mapError Http serverTask\n            , mapError WebGL textureTask\n            ]\n\n-}\nmapError : (x -> y) -> Task x a -> Task y a\nmapError convert task =\n    task\n        |> onError (fail << convert)\n\n\n\n-- COMMANDS\n\n\ntype MyCmd msg\n    = Perform (Task Never msg)\n    | Execute (Task Never {})\n\n\n{-| Like I was saying in the [`Task`](#Task) documentation, just having a\n`Task` does not mean it is done. We must command Gren to `perform` the task:\n\n\n\n    import Task\n    import Time\n\n    type Msg\n        = Click\n        | Search String\n        | NewTime Time.Posix\n\n    getNewTime : Cmd Msg\n    getNewTime =\n        Task.perform NewTime Time.now\n\nSo we have changed a task like \"make delicious lasagna\" into a command like\n\"Hey Gren, make delicious lasagna and give it to my `update` function as a\n`Msg` value.\"\n\n-}\nperform : (a -> msg) -> Task Never a -> Cmd msg\nperform toMessage task =\n    command (Perform (map toMessage task))\n\n\n{-| This is very similar to [`perform`](#perform) except it can handle failures!\nSo we could _attempt_ to focus on a certain DOM node like this:\n\n    -- gren install gren-lang/browser\n\n\n    import Browser.Dom\n    import Task\n\n    type Msg\n        = Click\n        | Search String\n        | Focus (Result Browser.DomError {})\n\n    focus : Cmd Msg\n    focus =\n        Task.attempt Focus (Browser.Dom.focus \"my-app-search-box\")\n\nSo the task is \"focus on this DOM node\" and we are turning it into the command\n\"Hey Gren, attempt to focus on this DOM node and give me a `Msg` about whether\nyou succeeded or failed.\"\n\n-}\nattempt : (Result x a -> msg) -> Task x a -> Cmd msg\nattempt resultToMessage task =\n    command\n        (Perform\n            (task\n                |> andThen (succeed << resultToMessage << Ok)\n                |> onError (succeed << resultToMessage << Err)\n            )\n        )\n\n\n{-| Sometimes we want to give a command without being told how it went. Maybe we\nare logging something to the screen, or changing the scroll position of the window.\nIn either case, there's really nothing for us to do afterwards. In those cases\nwe can use `execute`.\n-}\nexecute : Task Never a -> Cmd msg\nexecute task =\n    command (Execute (map (\\_ -> {}) task))\n\n\ncmdMap : (a -> b) -> MyCmd a -> MyCmd b\ncmdMap tagger cmd =\n    case cmd of\n        Perform task ->\n            Perform (map tagger task)\n\n        Execute task ->\n            Execute task\n\n\n-- MANAGER\n\n\ninit : Task Never {}\ninit =\n    succeed {}\n\n\nonEffects : Platform.Router msg Never -> Array (MyCmd msg) -> {} -> Task Never {}\nonEffects router commands state =\n    map\n        (\\_ -> {})\n        (sequence (Array.map (spawnCmd router) commands))\n\n\nonSelfMsg : Platform.Router msg Never -> Never -> {} -> Task Never {}\nonSelfMsg _ _ _ =\n    succeed {}\n\n\nspawnCmd : Platform.Router msg Never -> MyCmd msg -> Task x {}\nspawnCmd router cmd =\n    case cmd of\n        Perform task ->\n            Gren.Kernel.Scheduler.spawn\n                (task\n                    |> andThen (Platform.sendToApp router)\n                )\n\n        Execute task ->\n            Gren.Kernel.Scheduler.spawn task\n",
        "module Basics exposing\n    ( Int, (+), (-), (*), (/), (//), (^), negate\n    , Float, toFloat, isNaN, isInfinite\n    , (==), (/=)\n    , (<), (>), (<=), (>=), max, min, clamp, compare, Order(..)\n    , Bool(..), not, (&&), (||), xor\n    , (++)\n    , identity, (<|), (|>), (<<), (>>), Never, never\n    )\n\n{-| Tons of useful functions that get imported by default.\n\n\n## Numbers\n\n@docs Int, (+), (-), (*), (/), (//), (^), negate\n\n\n## Float\n\n@docs Float, toFloat, isNaN, isInfinite\n\n\n## Equality\n\n@docs (==), (/=)\n\n\n## Comparison\n\nThese functions only work on `comparable` types. This includes numbers,\ncharacters, strings and arrays of comparable things.\n\n@docs Order, (<), (>), (<=), (>=), max, min, clamp, compare\n\n\n## Booleans\n\n@docs Bool, not, (&&), (||), xor\n\n\n## Append Strings and Lists\n\n@docs (++)\n\n\n## Function Helpers\n\n@docs identity, (<|), (|>), (<<), (>>), Never, never\n\n-}\n\nimport Gren.Kernel.Basics\nimport Gren.Kernel.Utils\n\n\n\n-- INFIX OPERATORS\n\n\ninfix right 0 (<|) = apL\ninfix left  0 (|>) = apR\ninfix right 2 (||) = or\ninfix right 3 (&&) = and\ninfix non   4 (==) = eq\ninfix non   4 (/=) = neq\ninfix non   4 (<) = lt\ninfix non   4 (>) = gt\ninfix non   4 (<=) = le\ninfix non   4 (>=) = ge\ninfix right 5 (++) = append\ninfix left  6 (+) = add\ninfix left  6 (-) = sub\ninfix left  7 (*) = mul\ninfix left  7 (/) = fdiv\ninfix left  7 (//) = idiv\ninfix right 8 (^) = pow\ninfix left  9 (<<) = composeL\ninfix right 9 (>>) = composeR\n\n\n\n-- MATHEMATICS\n\n\n{-| An `Int` is a whole number. Valid syntax for integers includes:\n\n    0\n\n    42\n\n    9000\n\n    0xFF -- 255 in hexadecimal\n\n    0x0A --  10 in hexadecimal\n\n**Note:** `Int` math is well-defined in the range `-2^31` to `2^31 - 1`. Outside\nof that range, the behavior is determined by the compilation target. When\ngenerating JavaScript, the safe range expands to `-(2^53 - 1)` to `2^53 - 1` for some\noperations, but if we generate WebAssembly some day, we would do the traditional\n[integer overflow][io]. This quirk is necessary to get good performance on\nquirky compilation targets.\n\n**Historical Note:** The name `Int` comes from the term [integer]. It appears\nthat the `int` abbreviation was introduced in [ALGOL 68][68], shortening it\nfrom `integer` in [ALGOL 60][60]. Today, almost all programming languages use\nthis abbreviation.\n\n[io]: https://en.wikipedia.org/wiki/Integer_overflow\n[integer]: https://en.wikipedia.org/wiki/Integer\n[60]: https://en.wikipedia.org/wiki/ALGOL_60\n[68]: https://en.wikipedia.org/wiki/ALGOL_68\n\n-}\ntype Int\n    = Int -- NOTE: The compiler provides the real implementation.\n\n\n{-| A `Float` is a [floating-point number][fp]. Valid syntax for floats includes:\n\n    0\n    42\n    3.14\n    0.1234\n    6.022e23   -- == (6.022 * 10^23)\n    6.022e+23  -- == (6.022 * 10^23)\n    1.602e−19  -- == (1.602 * 10^-19)\n    1e3        -- == (1 * 10^3) == 1000\n\n**Historical Note:** The particular details of floats (e.g. `NaN`) are\nspecified by [IEEE 754][ieee] which is literally hard-coded into almost all\nCPUs in the world. That means if you think `NaN` is weird, you must\nsuccessfully overtake Intel and AMD with a chip that is not backwards\ncompatible with any widely-used assembly language.\n\n[fp]: https://en.wikipedia.org/wiki/Floating-point_arithmetic\n[ieee]: https://en.wikipedia.org/wiki/IEEE_754\n\n-}\ntype Float\n    = Float -- NOTE: The compiler provides the real implementation.\n\n\n{-| Add two numbers. The `number` type variable means this operation can be\nspecialized to `Int -> Int -> Int` or to `Float -> Float -> Float`. So you\ncan do things like this:\n\n    3002 + 4004 == 7006 -- all ints\n\n    3.14 + 3.14 == 6.28 -- all floats\n\nYou _cannot_ add an `Int` and a `Float` directly though. Use functions like\n[toFloat](#toFloat) or [round](#round) to convert both values to the same type.\nSo if you needed to add a list length to a `Float` for some reason, you\ncould say one of these:\n\n    3.14 + toFloat (List.length [ 1, 2, 3 ]) == 6.14\n\n    round 3.14 + List.length [ 1, 2, 3 ] == 6\n\n**Note:** Languages like Java and JavaScript automatically convert `Int` values\nto `Float` values when you mix and match. This can make it difficult to be sure\nexactly what type of number you are dealing with. When you try to _infer_ these\nconversions (as Scala does) it can be even more confusing. Gren has opted for a\ndesign that makes all conversions explicit.\n\n-}\nadd : number -> number -> number\nadd =\n    Gren.Kernel.Basics.add\n\n\n{-| Subtract numbers like `4 - 3 == 1`.\n\nSee [`(+)`](#+) for docs on the `number` type variable.\n\n-}\nsub : number -> number -> number\nsub =\n    Gren.Kernel.Basics.sub\n\n\n{-| Multiply numbers like `2 * 3 == 6`.\n\nSee [`(+)`](#+) for docs on the `number` type variable.\n\n-}\nmul : number -> number -> number\nmul =\n    Gren.Kernel.Basics.mul\n\n\n{-| Floating-point division:\n\n    10 / 4 == 2.5\n\n    11 / 4 == 2.75\n\n    12 / 4 == 3\n\n    13 / 4 == 3.25\n\n    14\n        / 4\n        == 3.5\n        - 1\n        / 4\n        == -0.25\n        - 5\n        / 4\n        == -1.25\n\n-}\nfdiv : Float -> Float -> Float\nfdiv =\n    Gren.Kernel.Basics.fdiv\n\n\n{-| Integer division:\n\n    10 // 4 == 2\n\n    11 // 4 == 2\n\n    12 // 4 == 3\n\n    13 // 4 == 3\n\n    14\n        // 4\n        == 3\n        - 1\n        // 4\n        == 0\n        - 5\n        // 4\n        == -1\n\nNotice that the remainder is discarded, so `3 // 4` is giving output\nsimilar to `truncate (3 / 4)`.\n\nIt may sometimes be useful to pair this with the [`remainderBy`](#remainderBy)\nfunction.\n\n-}\nidiv : Int -> Int -> Int\nidiv =\n    Gren.Kernel.Basics.idiv\n\n\n{-| Exponentiation\n\n    3 ^ 2 == 9\n\n    3 ^ 3 == 27\n\n-}\npow : number -> number -> number\npow =\n    Gren.Kernel.Basics.pow\n\n\n{-| Negate a number.\n\n    negate 42 == -42\n\n    negate -42 == 42\n\n    negate 0 == 0\n\n-}\nnegate : number -> number\nnegate n =\n    -n\n\n\n-- INT TO FLOAT / FLOAT TO INT\n\n\n{-| Convert an integer into a float. Useful when mixing `Int` and `Float`\nvalues like this:\n\n    halfOf : Int -> Float\n    halfOf number =\n        toFloat number / 2\n\n-}\ntoFloat : Int -> Float\ntoFloat =\n    Gren.Kernel.Basics.toFloat\n\n\n\n-- EQUALITY\n\n\n{-| Check if values are &ldquo;the same&rdquo;.\n\n**Note:** Gren uses structural equality on tuples, records, and user-defined\nunion types. This means the values `(3, 4)` and `(3, 4)` are definitely equal.\nThis is not true in languages like JavaScript that use reference equality on\nobjects.\n\n**Note:** Do not use `(==)` with functions, JSON values from `gren/json`, or\nregular expressions from `gren/regex`. It does not work. It will crash if\npossible. With JSON values, decode to Gren values before doing any equality\nchecks!\n\nWhy is it like this? Equality in the Gren sense can be difficult or impossible\nto compute. Proving that functions are the same is [undecidable], and JSON\nvalues can come in through ports and have functions, cycles, and new JS data\ntypes that interact weirdly with our equality implementation. In a future\nrelease, the compiler will detect when `(==)` is used with problematic types\nand provide a helpful error message at compile time. This will require some\npretty serious infrastructure work, so the stopgap is to crash as quickly as\npossible.\n\n[undecidable]: https://en.wikipedia.org/wiki/Undecidable_problem\n\n-}\neq : a -> a -> Bool\neq =\n    Gren.Kernel.Utils.equal\n\n\n{-| Check if values are not &ldquo;the same&rdquo;.\n\nSo `(a /= b)` is the same as `(not (a == b))`.\n\n-}\nneq : a -> a -> Bool\nneq =\n    Gren.Kernel.Utils.notEqual\n\n\n\n-- COMPARISONS\n\n\n{-| -}\nlt : comparable -> comparable -> Bool\nlt =\n    Gren.Kernel.Utils.lt\n\n\n{-| -}\ngt : comparable -> comparable -> Bool\ngt =\n    Gren.Kernel.Utils.gt\n\n\n{-| -}\nle : comparable -> comparable -> Bool\nle =\n    Gren.Kernel.Utils.le\n\n\n{-| -}\nge : comparable -> comparable -> Bool\nge =\n    Gren.Kernel.Utils.ge\n\n\n{-| Find the smaller of two comparables.\n\n    min 42 12345678 == 42\n\n    min \"abc\" \"xyz\" == \"abc\"\n\n-}\nmin : comparable -> comparable -> comparable\nmin x y =\n    if lt x y then\n        x\n\n    else\n        y\n\n\n{-| Find the larger of two comparables.\n\n    max 42 12345678 == 12345678\n\n    max \"abc\" \"xyz\" == \"xyz\"\n\n-}\nmax : comparable -> comparable -> comparable\nmax x y =\n    if gt x y then\n        x\n\n    else\n        y\n\n\n{-| Clamps a number within a given range. With the expression\n`clamp 100 200 x` the results are as follows:\n\n    100     if x < 100\n     x      if 100 <= x < 200\n    200     if 200 <= x\n\n-}\nclamp : number -> number -> number -> number\nclamp low high number =\n    if lt number low then\n        low\n\n    else if gt number high then\n        high\n\n    else\n        number\n\n\n{-| Compare any two comparable values. Comparable values include `String`,\n`Char`, `Int`, `Float`, or a list or tuple containing comparable values. These\nare also the only values that work as `Dict` keys or `Set` members.\n\n    compare 3 4 == LT\n\n    compare 4 4 == EQ\n\n    compare 5 4 == GT\n\n-}\ncompare : comparable -> comparable -> Order\ncompare =\n    Gren.Kernel.Utils.compare\n\n\n{-| Represents the relative ordering of two things.\nThe relations are less than, equal to, and greater than.\n-}\ntype Order\n    = LT\n    | EQ\n    | GT\n\n\n\n-- BOOLEANS\n\n\n{-| A “Boolean” value. It can either be `True` or `False`.\n\n**Note:** Programmers coming from JavaScript, Java, etc. tend to reach for\nboolean values way too often in Gren. Using a [union type][ut] is often clearer\nand more reliable. You can learn more about this from Jeremy [here][jf] or\nfrom Richard [here][rt].\n\n[ut]: https://guide.gren-lang.org/types/union_types.html\n[jf]: https://youtu.be/6TDKHGtAxeg?t=1m25s\n[rt]: https://youtu.be/IcgmSRJHu_8?t=1m14s\n\n-}\ntype Bool\n    = True\n    | False\n\n\n{-| Negate a boolean value.\n\n    not True == False\n\n    not False == True\n\n-}\nnot : Bool -> Bool\nnot =\n    Gren.Kernel.Basics.not\n\n\n{-| The logical AND operator. `True` if both inputs are `True`.\n\n    True && True == True\n\n    True && False == False\n\n    False && True == False\n\n    False && False == False\n\n**Note:** When used in the infix position, like `(left && right)`, the operator\nshort-circuits. This means if `left` is `False` we do not bother evaluating `right`\nand just return `False` overall.\n\n-}\nand : Bool -> Bool -> Bool\nand =\n    Gren.Kernel.Basics.and\n\n\n{-| The logical OR operator. `True` if one or both inputs are `True`.\n\n    True || True == True\n\n    True || False == True\n\n    False || True == True\n\n    False || False == False\n\n**Note:** When used in the infix position, like `(left || right)`, the operator\nshort-circuits. This means if `left` is `True` we do not bother evaluating `right`\nand just return `True` overall.\n\n-}\nor : Bool -> Bool -> Bool\nor =\n    Gren.Kernel.Basics.or\n\n\n{-| The exclusive-or operator. `True` if exactly one input is `True`.\n\n    xor True True == False\n\n    xor True False == True\n\n    xor False True == True\n\n    xor False False == False\n\n-}\nxor : Bool -> Bool -> Bool\nxor =\n    Gren.Kernel.Basics.xor\n\n\n\n-- APPEND\n\n\n{-| Put two appendable things together. This includes strings and lists.\n\n    \"hello\" ++ \"world\" == \"helloworld\"\n\n    [ 1, 1, 2 ] ++ [ 3, 5, 8 ] == [ 1, 1, 2, 3, 5, 8 ]\n\n-}\nappend : appendable -> appendable -> appendable\nappend =\n    Gren.Kernel.Utils.append\n\n\n\n-- CRAZY FLOATS\n\n\n{-| Determine whether a float is an undefined or unrepresentable number.\nNaN stands for _not a number_ and it is [a standardized part of floating point\nnumbers](https://en.wikipedia.org/wiki/NaN).\n\n    isNaN (0 / 0) == True\n\n    isNaN (sqrt -1) == True\n\n    isNaN (1 / 0) == False -- infinity is a number\n\n    isNaN 1 == False\n\n-}\nisNaN : Float -> Bool\nisNaN =\n    Gren.Kernel.Basics.isNaN\n\n\n{-| Determine whether a float is positive or negative infinity.\n\n    isInfinite (0 / 0) == False\n\n    isInfinite (sqrt -1) == False\n\n    isInfinite (1 / 0) == True\n\n    isInfinite 1 == False\n\nNotice that NaN is not infinite! For float `n` to be finite implies that\n`not (isInfinite n || isNaN n)` evaluates to `True`.\n\n-}\nisInfinite : Float -> Bool\nisInfinite =\n    Gren.Kernel.Basics.isInfinite\n\n\n\n-- FUNCTION HELPERS\n\n\n{-| Function composition, passing results along in the suggested direction. For\nexample, the following code checks if the result of rounding a float is odd:\n\n    not << isEven << round\n\nYou can think of this operator as equivalent to the following:\n\n    (g << f) == (\\x -> g (f x))\n\nSo our example expands out to something like this:\n\n    \\n -> not (isEven (round n))\n\n-}\ncomposeL : (b -> c) -> (a -> b) -> (a -> c)\ncomposeL g f x =\n    g (f x)\n\n\n{-| Function composition, passing results along in the suggested direction. For\nexample, the following code checks if the result of rounding a float is odd:\n\n    round >> isEven >> not\n\n-}\ncomposeR : (a -> b) -> (b -> c) -> (a -> c)\ncomposeR f g x =\n    g (f x)\n\n\n{-| Saying `x |> f` is exactly the same as `f x`.\n\nIt is called the “pipe” operator because it lets you write “pipelined” code.\nFor example, say we have a `sanitize` function for turning user input into\nintegers:\n\n    -- BEFORE\n    sanitize : String -> Maybe Int\n    sanitize input =\n        String.toInt (String.trim input)\n\nWe can rewrite it like this:\n\n    -- AFTER\n    sanitize : String -> Maybe Int\n    sanitize input =\n        input\n            |> String.trim\n            |> String.toInt\n\nTotally equivalent! I recommend trying to rewrite code that uses `x |> f`\ninto code like `f x` until there are no pipes left. That can help you build\nyour intuition.\n\n**Note:** This can be overused! I think folks find it quite neat, but when you\nhave three or four steps, the code often gets clearer if you break out a\ntop-level helper function. Now the transformation has a name. The arguments are\nnamed. It has a type annotation. It is much more self-documenting that way!\nTesting the logic gets easier too. Nice side benefit!\n\n-}\napR : a -> (a -> b) -> b\napR x f =\n    f x\n\n\n{-| Saying `f <| x` is exactly the same as `f x`.\n\nIt can help you avoid parentheses, which can be nice sometimes. Maybe you want\nto apply a function to a `case` expression? That sort of thing.\n\n-}\napL : (a -> b) -> a -> b\napL f x =\n    f x\n\n\n{-| Given a value, returns exactly the same value. This is called\n[the identity function](https://en.wikipedia.org/wiki/Identity_function).\n-}\nidentity : a -> a\nidentity x =\n    x\n\n\n{-| A value that can never happen! For context:\n\n  - The boolean type `Bool` has two values: `True` and `False`\n  - The unit type `()` has one value: `()`\n  - The never type `Never` has no values!\n\nYou may see it in the wild in `Html Never` which means this HTML will never\nproduce any messages. You would need to write an event handler like\n`onClick ??? : Attribute Never` but how can we fill in the question marks?!\nSo there cannot be any event handlers on that HTML.\n\nYou may also see this used with tasks that never fail, like `Task Never ()`.\n\nThe `Never` type is useful for restricting _arguments_ to a function. Maybe my\nAPI can only accept HTML without event handlers, so I require `Html Never` and\nusers can give `Html msg` and everything will go fine. Generally speaking, you\ndo not want `Never` in your return types though.\n\n-}\ntype Never\n    = JustOneMore Never\n\n\n{-| A function that can never be called. Seems extremely pointless, but it\n_can_ come in handy. Imagine you have some HTML that should never produce any\nmessages. And say you want to use it in some other HTML that _does_ produce\nmessages. You could say:\n\n    import Html exposing (..)\n\n    embedHtml : Html Never -> Html msg\n    embedHtml staticStuff =\n        div []\n            [ text \"hello\"\n            , Html.map never staticStuff\n            ]\n\nSo the `never` function is basically telling the type system, make sure no one\never calls me!\n\n-}\nnever : Never -> a\nnever (JustOneMore nvr) =\n    never nvr\n",
        "module Dict exposing\n    ( Dict\n    , empty, singleton, set, update, remove\n    , isEmpty, count, get, member, first, last, findFirst, findLast, any, all\n    , keys, values\n    , map, foldl, foldr, filter, filterMap, partition\n    , union, intersect, diff, merge\n    )\n\n{-| A dictionary mapping unique keys to values. The keys can be any comparable\ntype. This includes `Int`, `Float`, `Time`, `Char` and `String`.\n\nSet, remove, and query operations all take _O(log n)_ time.\n\n\n@docs Dict\n\n\n@docs empty, singleton, set, update, remove\n\n\n## Query\n\n@docs isEmpty, count, get, member, first, last, findFirst, findLast, any, all\n\n\n## Arrays\n\n@docs keys, values\n\n\n## Transform\n\n@docs map, foldl, foldr, filter, filterMap, partition\n\n\n## Combine\n\n@docs union, intersect, diff, merge\n\n-}\n\n\nimport Array exposing (Array)\nimport Basics exposing (..)\nimport Maybe exposing (..)\n\n\n\n-- DICTIONARIES\n-- The color of a node. Leaves are considered Black.\n\n\ntype NColor\n    = Red\n    | Black\n\n\n{-| A dictionary of keys and values. So a `Dict String User` is a dictionary\nthat lets you look up a `String` (such as user names) and find the associated\n`User`.\n\n    import Dict exposing ( Dict )\n\n    users : Dict String User\n    users =\n        Dict.fromArray\n            [ { key = \"Alice\"\n              , value = makeUser \"Alice\" 28 1.65\n              }\n            , { key = \"Bob\"\n              , value = makeUser \"Bob\" 19 1.82\n              }\n            , { key = \"Chuck\"\n              , value = makeUser \"Chuck\" 33 1.75\n              }\n            ]\n\n    type alias User =\n        { name : String\n        , age : Int\n        , height : Float\n        }\n\n    makeUser : String -> Int -> Float -> User\n    makeUser name age height =\n        { name = name\n        , age = age\n        , height = height\n        }\n-}\ntype Dict k v\n    = RBNode_gren_builtin NColor k v (Dict k v) (Dict k v)\n    | RBEmpty_gren_builtin\n\n\n{-| Create an empty dictionary.\n-}\nempty : Dict k v\nempty =\n    RBEmpty_gren_builtin\n\n\n{-| Get the value associated with a key. If the key is not found, return\n`Nothing`. This is useful when you are not sure if a key will be in the\ndictionary.\n\n    animals = fromArray [ (\"Tom\", Cat), (\"Jerry\", Mouse) ]\n\n    get \"Tom\"   animals == Just Cat\n    get \"Jerry\" animals == Just Mouse\n    get \"Spike\" animals == Nothing\n\n-}\nget : comparable -> Dict comparable v -> Maybe v\nget targetKey dict =\n    case dict of\n        RBEmpty_gren_builtin ->\n            Nothing\n\n        RBNode_gren_builtin _ key value left right ->\n            case compare targetKey key of\n                LT ->\n                    get targetKey left\n\n                EQ ->\n                    Just value\n\n                GT ->\n                    get targetKey right\n\n\n{-| Determine if a key is in a dictionary.\n-}\nmember : comparable -> Dict comparable v -> Bool\nmember key dict =\n    case get key dict of\n        Just _ ->\n            True\n\n        Nothing ->\n            False\n\n\n{-| Determine the number of key-value pairs in the dictionary.\n-}\ncount : Dict k v -> Int\ncount dict =\n    countHelp 0 dict\n\n\ncountHelp : Int -> Dict k v -> Int\ncountHelp n dict =\n    case dict of\n        RBEmpty_gren_builtin ->\n            n\n\n        RBNode_gren_builtin _ _ _ left right ->\n            countHelp (countHelp (n + 1) right) left\n\n\n{-| Retrieve the first, or lowest, key-value pair.\n-}\nfirst : Dict k v -> Maybe { key : k, value : v }\nfirst dict =\n    case dict of\n        RBEmpty_gren_builtin ->\n            Nothing\n\n        RBNode_gren_builtin _ k v RBEmpty_gren_builtin _ ->\n            Just { key = k, value = v }\n\n        RBNode_gren_builtin _ _ _ left _ ->\n            first left\n\n\n{-| Retrieve the last, or highest, key-value pair.\n-}\nlast : Dict k v -> Maybe { key : k, value : v }\nlast dict =\n    case dict of\n        RBEmpty_gren_builtin ->\n            Nothing\n\n        RBNode_gren_builtin _ k v _ RBEmpty_gren_builtin ->\n            Just { key = k, value = v }\n\n        RBNode_gren_builtin _ _ _ _ right ->\n            last right\n\n\n{-| Find the first key-value pair that passes the test.\n-}\nfindFirst : (k -> v -> Bool) -> Dict k v -> Maybe { key : k, value : v }\nfindFirst fn dict =\n    case dict of\n        RBEmpty_gren_builtin ->\n            Nothing\n\n        RBNode_gren_builtin _ key value left right ->\n            case findFirst fn left of\n                Nothing ->\n                    if fn key value then\n                        Just { key = key, value = value }\n\n                    else\n                        findFirst fn right\n\n                foundValue ->\n                    foundValue\n\n\n{-| Find the last key-value pair that passes the test.\n-}\nfindLast : (k -> v -> Bool) -> Dict k v -> Maybe { key : k, value : v }\nfindLast fn dict =\n    case dict of\n        RBEmpty_gren_builtin ->\n            Nothing\n\n        RBNode_gren_builtin _ key value left right ->\n            case findLast fn right of\n                Nothing ->\n                    if fn key value then\n                        Just { key = key, value = value }\n\n                    else\n                        findLast fn left\n\n                foundValue ->\n                    foundValue\n\n\n{-| Checks if any key-value pair in the dictionary passes the test.\n-}\nany : (k -> v -> Bool) -> Dict k v -> Bool\nany fn dict =\n    case findFirst fn dict of\n        Just _ ->\n            True\n\n        Nothing ->\n            False\n\n\n{-| Checks if all key-value pairs in the dictionary passes the test.\n-}\nall : (k -> v -> Bool) -> Dict k v -> Bool\nall fn dict =\n    case findFirst (\\key value -> not <| fn key value) dict of\n        Just _ ->\n            False\n\n        Nothing ->\n            True\n\n\n{-| Determine if a dictionary is empty.\n\n    isEmpty empty == True\n\n-}\nisEmpty : Dict k v -> Bool\nisEmpty dict =\n    case dict of\n        RBEmpty_gren_builtin ->\n            True\n\n        RBNode_gren_builtin _ _ _ _ _ ->\n            False\n\n\n{-| Sets a value for a given key. Existing values will be replaced.\nIf the key isn't already registered, the key-value pair will be inserted.\n-}\nset : comparable -> v -> Dict comparable v -> Dict comparable v\nset key value dict =\n    -- Root node is always Black\n    case setHelp key value dict of\n        RBNode_gren_builtin Red k v l r ->\n            RBNode_gren_builtin Black k v l r\n\n        x ->\n            x\n\n\nsetHelp : comparable -> v -> Dict comparable v -> Dict comparable v\nsetHelp key value dict =\n    case dict of\n        RBEmpty_gren_builtin ->\n            -- New nodes are always red. If it violates the rules, it will be fixed\n            -- when balancing.\n            RBNode_gren_builtin Red key value RBEmpty_gren_builtin RBEmpty_gren_builtin\n\n        RBNode_gren_builtin nColor nKey nValue nLeft nRight ->\n            case compare key nKey of\n                LT ->\n                    balance nColor nKey nValue (setHelp key value nLeft) nRight\n\n                EQ ->\n                    RBNode_gren_builtin nColor nKey value nLeft nRight\n\n                GT ->\n                    balance nColor nKey nValue nLeft (setHelp key value nRight)\n\n\nbalance : NColor -> k -> v -> Dict k v -> Dict k v -> Dict k v\nbalance color key value left right =\n    case right of\n        RBNode_gren_builtin Red rK rV rLeft rRight ->\n            case left of\n                RBNode_gren_builtin Red lK lV lLeft lRight ->\n                    RBNode_gren_builtin\n                        Red\n                        key\n                        value\n                        (RBNode_gren_builtin Black lK lV lLeft lRight)\n                        (RBNode_gren_builtin Black rK rV rLeft rRight)\n\n                _ ->\n                    RBNode_gren_builtin color rK rV (RBNode_gren_builtin Red key value left rLeft) rRight\n\n        _ ->\n            case left of\n                RBNode_gren_builtin Red lK lV (RBNode_gren_builtin Red llK llV llLeft llRight) lRight ->\n                    RBNode_gren_builtin\n                        Red\n                        lK\n                        lV\n                        (RBNode_gren_builtin Black llK llV llLeft llRight)\n                        (RBNode_gren_builtin Black key value lRight right)\n\n                _ ->\n                    RBNode_gren_builtin color key value left right\n\n\n{-| Remove a key-value pair from a dictionary. If the key is not found,\nno changes are made.\n-}\nremove : comparable -> Dict comparable v -> Dict comparable v\nremove key dict =\n    -- Root node is always Black\n    case removeHelp key dict of\n        RBNode_gren_builtin Red k v l r ->\n            RBNode_gren_builtin Black k v l r\n\n        x ->\n            x\n\n\n{-| The easiest thing to remove from the tree, is a red node. However, when searching for the\nnode to remove, we have no way of knowing if it will be red or not. This remove implementation\nmakes sure that the bottom node is red by moving red colors down the tree through rotation\nand color flips. Any violations this will cause, can easily be fixed by balancing on the way\nup again.\n-}\nremoveHelp : comparable -> Dict comparable v -> Dict comparable v\nremoveHelp targetKey dict =\n    case dict of\n        RBEmpty_gren_builtin ->\n            RBEmpty_gren_builtin\n\n        RBNode_gren_builtin color key value left right ->\n            if targetKey < key then\n                case left of\n                    RBNode_gren_builtin Black _ _ lLeft _ ->\n                        case lLeft of\n                            RBNode_gren_builtin Red _ _ _ _ ->\n                                RBNode_gren_builtin color key value (removeHelp targetKey left) right\n\n                            _ ->\n                                case moveRedLeft dict of\n                                    RBNode_gren_builtin nColor nKey nValue nLeft nRight ->\n                                        balance nColor nKey nValue (removeHelp targetKey nLeft) nRight\n\n                                    RBEmpty_gren_builtin ->\n                                        RBEmpty_gren_builtin\n\n                    _ ->\n                        RBNode_gren_builtin color key value (removeHelp targetKey left) right\n\n            else\n                removeHelpEQGT targetKey (removeHelpPrepEQGT targetKey dict color key value left right)\n\n\nremoveHelpPrepEQGT : comparable -> Dict comparable v -> NColor -> comparable -> v -> Dict comparable v -> Dict comparable v -> Dict comparable v\nremoveHelpPrepEQGT targetKey dict color key value left right =\n    case left of\n        RBNode_gren_builtin Red lK lV lLeft lRight ->\n            RBNode_gren_builtin\n                color\n                lK\n                lV\n                lLeft\n                (RBNode_gren_builtin Red key value lRight right)\n\n        _ ->\n            case right of\n                RBNode_gren_builtin Black _ _ (RBNode_gren_builtin Black _ _ _ _) _ ->\n                    moveRedRight dict\n\n                RBNode_gren_builtin Black _ _ RBEmpty_gren_builtin _ ->\n                    moveRedRight dict\n\n                _ ->\n                    dict\n\n\n{-| When we find the node we are looking for, we can remove by replacing the key-value\npair with the key-value pair of the left-most node on the right side (the closest pair).\n-}\nremoveHelpEQGT : comparable -> Dict comparable v -> Dict comparable v\nremoveHelpEQGT targetKey dict =\n    case dict of\n        RBNode_gren_builtin color key value left right ->\n            if targetKey == key then\n                case getMin right of\n                    RBNode_gren_builtin _ minKey minValue _ _ ->\n                        balance color minKey minValue left (removeMin right)\n\n                    RBEmpty_gren_builtin ->\n                        RBEmpty_gren_builtin\n\n            else\n                balance color key value left (removeHelp targetKey right)\n\n        RBEmpty_gren_builtin ->\n            RBEmpty_gren_builtin\n\n\ngetMin : Dict k v -> Dict k v\ngetMin dict =\n    case dict of\n        RBNode_gren_builtin _ _ _ ((RBNode_gren_builtin _ _ _ _ _) as left) _ ->\n            getMin left\n\n        _ ->\n            dict\n\n\nremoveMin : Dict k v -> Dict k v\nremoveMin dict =\n    case dict of\n        RBNode_gren_builtin color key value ((RBNode_gren_builtin lColor _ _ lLeft _) as left) right ->\n            case lColor of\n                Black ->\n                    case lLeft of\n                        RBNode_gren_builtin Red _ _ _ _ ->\n                            RBNode_gren_builtin color key value (removeMin left) right\n\n                        _ ->\n                            case moveRedLeft dict of\n                                RBNode_gren_builtin nColor nKey nValue nLeft nRight ->\n                                    balance nColor nKey nValue (removeMin nLeft) nRight\n\n                                RBEmpty_gren_builtin ->\n                                    RBEmpty_gren_builtin\n\n                _ ->\n                    RBNode_gren_builtin color key value (removeMin left) right\n\n        _ ->\n            RBEmpty_gren_builtin\n\n\nmoveRedLeft : Dict k v -> Dict k v\nmoveRedLeft dict =\n    case dict of\n        RBNode_gren_builtin clr k v (RBNode_gren_builtin lClr lK lV lLeft lRight) (RBNode_gren_builtin rClr rK rV ((RBNode_gren_builtin Red rlK rlV rlL rlR) as rLeft) rRight) ->\n            RBNode_gren_builtin\n                Red\n                rlK\n                rlV\n                (RBNode_gren_builtin Black k v (RBNode_gren_builtin Red lK lV lLeft lRight) rlL)\n                (RBNode_gren_builtin Black rK rV rlR rRight)\n\n        RBNode_gren_builtin clr k v (RBNode_gren_builtin lClr lK lV lLeft lRight) (RBNode_gren_builtin rClr rK rV rLeft rRight) ->\n            case clr of\n                Black ->\n                    RBNode_gren_builtin\n                        Black\n                        k\n                        v\n                        (RBNode_gren_builtin Red lK lV lLeft lRight)\n                        (RBNode_gren_builtin Red rK rV rLeft rRight)\n\n                Red ->\n                    RBNode_gren_builtin\n                        Black\n                        k\n                        v\n                        (RBNode_gren_builtin Red lK lV lLeft lRight)\n                        (RBNode_gren_builtin Red rK rV rLeft rRight)\n\n        _ ->\n            dict\n\n\nmoveRedRight : Dict k v -> Dict k v\nmoveRedRight dict =\n    case dict of\n        RBNode_gren_builtin clr k v (RBNode_gren_builtin lClr lK lV (RBNode_gren_builtin Red llK llV llLeft llRight) lRight) (RBNode_gren_builtin rClr rK rV rLeft rRight) ->\n            RBNode_gren_builtin\n                Red\n                lK\n                lV\n                (RBNode_gren_builtin Black llK llV llLeft llRight)\n                (RBNode_gren_builtin Black k v lRight (RBNode_gren_builtin Red rK rV rLeft rRight))\n\n        RBNode_gren_builtin clr k v (RBNode_gren_builtin lClr lK lV lLeft lRight) (RBNode_gren_builtin rClr rK rV rLeft rRight) ->\n            case clr of\n                Black ->\n                    RBNode_gren_builtin\n                        Black\n                        k\n                        v\n                        (RBNode_gren_builtin Red lK lV lLeft lRight)\n                        (RBNode_gren_builtin Red rK rV rLeft rRight)\n\n                Red ->\n                    RBNode_gren_builtin\n                        Black\n                        k\n                        v\n                        (RBNode_gren_builtin Red lK lV lLeft lRight)\n                        (RBNode_gren_builtin Red rK rV rLeft rRight)\n\n        _ ->\n            dict\n\n\n{-| Update the value of a dictionary for a specific key with a given function.\n-}\nupdate : comparable -> (Maybe v -> Maybe v) -> Dict comparable v -> Dict comparable v\nupdate targetKey alter dictionary =\n    case alter (get targetKey dictionary) of\n        Just value ->\n            set targetKey value dictionary\n\n        Nothing ->\n            remove targetKey dictionary\n\n\n{-| Create a dictionary with one key-value pair.\n-}\nsingleton : comparable -> v -> Dict comparable v\nsingleton key value =\n    -- Root node is always Black\n    RBNode_gren_builtin Black key value RBEmpty_gren_builtin RBEmpty_gren_builtin\n\n\n\n-- COMBINE\n\n\n{-| Combine two dictionaries. If there is a collision, preference is given\nto the first dictionary.\n-}\nunion : Dict comparable v -> Dict comparable v -> Dict comparable v\nunion t1 t2 =\n    foldl set t2 t1\n\n\n{-| Keep a key-value pair when its key appears in the second dictionary.\nPreference is given to values in the first dictionary.\n-}\nintersect : Dict comparable v -> Dict comparable v -> Dict comparable v\nintersect t1 t2 =\n    filter (\\k _ -> member k t2) t1\n\n\n{-| Keep a key-value pair when its key does not appear in the second dictionary.\n-}\ndiff : Dict comparable a -> Dict comparable b -> Dict comparable a\ndiff t1 t2 =\n    foldl (\\k v t -> remove k t) t1 t2\n\n\n\n-- TRANSFORM\n\n\n{-| Apply a function to all values in a dictionary.\n-}\nmap : (k -> a -> b) -> Dict k a -> Dict k b\nmap func dict =\n    case dict of\n        RBEmpty_gren_builtin ->\n            RBEmpty_gren_builtin\n\n        RBNode_gren_builtin color key value left right ->\n            RBNode_gren_builtin color key (func key value) (map func left) (map func right)\n\n\n{-| Fold over the key-value pairs in a dictionary from lowest key to highest key.\n\n    import Dict exposing (Dict)\n\n    getAges : Dict String User -> Array String\n    getAges users =\n        Dict.foldl addAge [] users\n\n    addAge : String -> User -> Array String -> Array String\n    addAge _ user ages =\n        user.age :: ages\n\n    -- getAges users == [33,19,28]\n\n-}\nfoldl : (k -> v -> b -> b) -> b -> Dict k v -> b\nfoldl func acc dict =\n    case dict of\n        RBEmpty_gren_builtin ->\n            acc\n\n        RBNode_gren_builtin _ key value left right ->\n            foldl func (func key value (foldl func acc left)) right\n\n\n{-| Fold over the key-value pairs in a dictionary from highest key to lowest key.\n\n    import Dict exposing (Dict)\n\n    getAges : Dict String User -> Array String\n    getAges users =\n        Dict.foldr addAge [] users\n\n    addAge : String -> User -> Array String -> Array String\n    addAge _ user ages =\n        user.age :: ages\n\n    -- getAges users == [28,19,33]\n\n-}\nfoldr : (k -> v -> b -> b) -> b -> Dict k v -> b\nfoldr func acc t =\n    case t of\n        RBEmpty_gren_builtin ->\n            acc\n\n        RBNode_gren_builtin _ key value left right ->\n            foldr func (func key value (foldr func acc right)) left\n\n\n{-| Keep only the key-value pairs that pass the given test.\n-}\nfilter : (comparable -> v -> Bool) -> Dict comparable v -> Dict comparable v\nfilter isGood dict =\n    foldl\n        (\\k v d ->\n            if isGood k v then\n                set k v d\n\n            else\n                d\n        )\n        empty\n        dict\n\n\n{-| Filter out unwanted results of a map operation.\n-}\nfilterMap : (comparable -> v -> Maybe x) -> Dict comparable v -> Dict comparable x\nfilterMap toMaybe dict =\n    foldl\n        (\\k v d ->\n            case toMaybe k v of\n                Just newValue ->\n                    set k newValue d\n\n                Nothing ->\n                    d\n        )\n        empty\n        dict\n\n\n{-| Partition a dictionary according to some test. The first dictionary\ncontains all key-value pairs which passed the test, and the second contains\nthe pairs that did not.\n-}\npartition : (comparable -> v -> Bool) -> Dict comparable v -> { trues : Dict comparable v, falses : Dict comparable v }\npartition isGood dict =\n    let\n        add key value { trues, falses } =\n            if isGood key value then\n                { trues = set key value trues\n                , falses = falses\n                }\n\n            else\n                { trues = trues\n                , falses = set key value falses\n                }\n    in\n    foldl add { trues = empty, falses = empty } dict\n\n\n\n-- LISTS\n\n\n{-| Get all of the keys in a dictionary, sorted from lowest to highest.\n\n    keys (Dict.empty |> Dict.set 0 \"Alice\" |> Dict.set 1 \"Bob\") == [ 0, 1 ]\n\n-}\nkeys : Dict k v -> Array k\nkeys dict =\n    foldl (\\key value keyArray -> Array.pushLast key keyArray) [] dict\n\n\n{-| Get all of the values in a dictionary, in the order of their keys.\n\n    values (Dict.empty |> Dict.set 0 \"Alice\" |> Dict.set 1 \"Bob\") == [ \"Alice\", \"Bob\" ]\n\n-}\nvalues : Dict k v -> Array v\nvalues dict =\n    foldl (\\key value valueArray -> Array.pushLast value valueArray) [] dict\n\n\n{-| The most general way of combining two dictionaries. You provide three\naccumulators for when a given key appears:\n\n1.  Only in the left dictionary.\n2.  In both dictionaries.\n3.  Only in the right dictionary.\n    You then traverse all the keys from lowest to highest, building up whatever\n    you want.\n\n-}\nmerge :\n    (comparable -> a -> result -> result)\n    -> (comparable -> a -> b -> result -> result)\n    -> (comparable -> b -> result -> result)\n    -> Dict comparable a\n    -> Dict comparable b\n    -> result\n    -> result\nmerge leftStep bothStep rightStep leftDict rightDict initialResult =\n    let\n        stepState rKey rValue { list, result } =\n            case Array.popFirst list of\n                Nothing ->\n                    { list = list\n                    , result = rightStep rKey rValue result\n                    }\n\n                Just { first = { key = lKey, value = lValue }, rest } ->\n                    if lKey < rKey then\n                        stepState rKey rValue \n                            { list = rest\n                            , result = leftStep lKey lValue result\n                            }\n\n                    else if lKey > rKey then\n                        { list = list\n                        , result = rightStep rKey rValue result\n                        }\n\n                    else\n                        { list = rest\n                        , result = bothStep lKey lValue rValue result\n                        }\n\n        { list = leftovers, result = intermediateResult } =\n            foldl stepState { list = foldl (\\key value array -> Array.pushLast { key = key, value = value } array) [] leftDict, result = initialResult } rightDict\n    in\n    Array.foldl (\\{ key, value } result -> leftStep key value result) intermediateResult leftovers\n",
        "module Array exposing\n    ( Array\n    , empty, singleton, initialize, repeat, range\n    , map, indexedMap, foldl, foldr, indexedFoldl, indexedFoldr, filter, filterMap, reverse\n    , isEmpty, length, get, findFirst, findLast, member, any, all, minimum, maximum\n    , set, update, pushFirst, pushLast\n    , prepend, append, flatten, flatMap, intersperse, map2, map3\n    , first, last, slice, dropFirst, dropLast, takeFirst, takeLast, popFirst, popLast, partition\n    , sort, sortBy, sortWith\n    )\n\n{-| You can create an `Array` using the `[1, 2, 3]` syntax. This module has a bunch of\nfunctions to help you work with them.\n\n@docs Array\n\n\n@docs empty, singleton, initialize, repeat, range\n\n\n## Transform\n\n@docs map, indexedMap, foldl, foldr, indexedFoldl, indexedFoldr, filter, filterMap, reverse\n\n\n## Query\n\n@docs isEmpty, length, get, findFirst, findLast, member, any, all, minimum, maximum\n\n\n## Modify\n\n@docs set, update, pushFirst, pushLast\n\n\n## Combine\n\n@docs prepend, append, flatten, flatMap, intersperse, map2, map3\n\n\n## Deconstruct\n\n@docs slice, dropFirst, dropLast, takeFirst, takeLast, popFirst, popLast, first, last, partition\n\n\n## Sort\n\n@docs sort, sortBy, sortWith\n\n\n-}\n\nimport Basics exposing (..)\nimport Maybe exposing (Maybe(..))\nimport Gren.Kernel.Array\n\n\n{-| An Array is an ordered collection of elements.\n-}\ntype Array a\n    = Array a\n\n\n-- CREATE\n\n\n{-| An empty array.\n-}\nempty : Array a\nempty =\n    []\n\n\n{-| Create an array containing a single value.\n-}\nsingleton : a -> Array a\nsingleton a =\n    [ a ]\n\n\n{-| Create an array of `n` elements, containing the elements\nresulting from calling `fn` with `offset + index`.\n\n    initialize 3 5 identity == [ 5, 6, 7 ]\n\nIn the above example, we create an array containing 3 integers\nstarting at 5.\n-}\ninitialize : Int -> Int -> (Int -> a) -> Array a\ninitialize =\n    Gren.Kernel.Array.initialize\n\n\n{-| Create an array with `n` copies of a value:\n\n    repeat 5 3 == [ 3, 3, 3, 3, 3 ]\n\n-}\nrepeat : Int -> a -> Array a\nrepeat n val =\n    initialize n 0 (\\_ -> val)\n\n\n{-| Create a list of numbers, every element increasing by one. You give the lowest and highest number that should be in the list.\n\n    range 3 6 == [3, 4, 5, 6]\n    range 3 3 == [3]\n    range 6 3 == []\n\n-}\nrange : Int -> Int -> Array Int\nrange from to =\n    if from > to then\n        []\n\n    else if from == to then\n        [from]\n\n    else \n        initialize (to - from + 1) from identity\n\n\n-- TRANSFORM\n\n\n{-| Apply a function on every element in an array.\n\n    map negate [ 1, 4, 9 ] == [ -1, -4, -9 ]\n\nSo `map func [ a, b, c ]` is the same as `[ func a, func b, func c ]`\n\n-}\nmap : (a -> b) -> Array a -> Array b\nmap =\n    Gren.Kernel.Array.map\n\n\n{-| Same as `map` but the function is also applied to the index of each element.\n\n    indexedMap (\\idx val -> [idx, val]) [ 3, 3, 3 ] == [ [ 0, 3 ], [ 1, 3 ], [ 2, 3 ] ]\n\n-}\nindexedMap : (Int -> a -> b) -> Array a -> Array b\nindexedMap =\n    Gren.Kernel.Array.indexedMap\n\n\n{-| Reduce the array from the left.\n\n    foldl (+) 0 [ 1, 2, 3 ] == 6\n\nSo `foldl step state [ 1, 2, 3 ]` is like saying:\n\n    state\n        |> step 1\n        |> step 2\n        |> step 3\n-}\nfoldl : (a -> b -> b) -> b -> Array a -> b\nfoldl =\n    Gren.Kernel.Array.foldl\n\n\n{-| Reduce the array from the right. Same as `foldl` but\nthe execution order is reversed.\n-}\nfoldr : (a -> b -> b) -> b -> Array a -> b\nfoldr =\n    Gren.Kernel.Array.foldr\n\n\n{-| Reduce the array from the left. The reducing function is \npassed the index of the current value.\n\n    indexedFoldl (\\idx val sum -> idx + val + sum) 0 [ 1, 2, 3 ] == 9\n\n-}\nindexedFoldl : (Int -> a -> b -> b) -> b -> Array a -> b\nindexedFoldl =\n    Gren.Kernel.Array.indexedFoldl\n\n\n{-| Reduce the array from the right. The reducing function\nis passed the index of the current value. Same as `indexedFoldl`\nbut the execution order is reversed.\n-}\nindexedFoldr : (Int -> a -> b -> b) -> b -> Array a -> b\nindexedFoldr =\n    Gren.Kernel.Array.indexedFoldr\n\n\n{-| Keep values that pass the test.\n\n    filter (\\n -> n < 3) [ 1, 2, 3, 4 ] == [ 1, 2 ]\n\n-}\nfilter : (a -> Bool) -> Array a -> Array a\nfilter =\n    Gren.Kernel.Array.filter\n\n\n{-| Filter out unwanted results of a map operation.\n\n    filterMap String.toInt [ \"3\", \"not a number\", \"-5\" ] == [ 3, -5 ]\n    filterMap identity [ Just 1, Nothing ] == [ 1 ]\n\n-}\nfilterMap : (a -> Maybe b) -> Array a -> Array b\nfilterMap mapper array =\n    flatMap\n        (\\v ->\n            case mapper v of\n                Just newValue ->\n                    [ newValue ]\n\n                Nothing ->\n                    []\n        )\n        array\n\n\n{-| Reverse an array.\n\n    reverse [ 1, 2, 3 ] == [ 3, 2, 1 ]\n\n-}\nreverse : Array a -> Array a\nreverse =\n    Gren.Kernel.Array.reverse\n\n\n-- QUERY\n\n\n{-| Check if an array is empty.\n\n    isEmpty [] == True\n    isEmpty [ 1, 2, 3 ] == False\n\n-}\nisEmpty : Array a -> Bool\nisEmpty array =\n    length array == 0\n\n\n{-| Return the length of an array.\n\n    length [ 1, 2, 3 ] == 3\n\n-}\nlength : Array a -> Int\nlength =\n    Gren.Kernel.Array.length\n\n\n{-| Retrieve the element at a given index, or `Nothing` if the index is out of bounds.\nA negative index looks up an element in reverse from the end of the array.\n\n    get 1 [ 1, 2, 3 ] == Just 2\n    get 10 [ 1, 2, 3 ] == Nothing\n    get -1 [ 1, 2, 3 ] == Just 3\n\n-}\nget : Int -> Array a -> Maybe a\nget =\n    Gren.Kernel.Array.get\n\n\n{-| Find the first value that passes the test.\n\n    find (\\n -> n > 0) [ -1, 0, 1, 2 ] == Just 1\n\n-}\nfindFirst : (a -> Bool) -> Array a -> Maybe a\nfindFirst =\n    Gren.Kernel.Array.findFirst\n\n\n{-| Find the last value that passes the test.\n\n    find (\\n -> n > 0) [ -1, 0, 1, 2 ] == Just 2\n\n-}\nfindLast : (a -> Bool) -> Array a -> Maybe a\nfindLast =\n    Gren.Kernel.Array.findLast\n\n\n{-| Figure out whether an array contains a value.\n\n    member 9 [1,2,3,4] == False\n    member 4 [1,2,3,4] == True\n\n-}\nmember : a -> Array a -> Bool\nmember value array =\n    case findFirst (\\v -> v == value) array of\n        Just _ ->\n            True\n\n        Nothing ->\n            False\n\n\n{-| Determine if any elements pass the test.\n\n    any isEven [2,3] == True\n    any isEven [1,3] == False\n    any isEven [] == False\n\n-}\nany : (a -> Bool) -> Array a -> Bool\nany fn array =\n    case findFirst fn array of\n        Just _ ->\n            True\n\n        Nothing ->\n            False\n\n\n{-| Determine if all elements pass the test.\n\n    all isEven [2,4] == True\n    all isEven [2,3] == False\n    all isEven [] == True\n\n-}\nall : (a -> Bool) -> Array a -> Bool\nall fn array =\n    case findFirst (not << fn) array of\n        Just _ ->\n            False\n\n        Nothing ->\n            True\n\n\n{-| Find the minimum element in a non-empty list.\n\n    minimum [3,2,1] == Just 1\n    minimum []      == Nothing\n\n-}\nminimum : Array comparable -> Maybe comparable\nminimum array =\n    case first array of\n        Nothing ->\n            Nothing\n\n        Just val ->\n            Just <|\n                foldl\n                    (\\current lowest ->\n                        if current < lowest then\n                            current\n\n                        else\n                            lowest\n                    )\n                    val\n                    array\n\n\n{-| Find the maximum element in a non-empty list.\n\n    maximum [3,2,1] == Just 3\n    maximum []      == Nothing\n\n-}\nmaximum : Array comparable -> Maybe comparable\nmaximum array =\n    case first array of\n        Nothing ->\n            Nothing\n\n        Just val ->\n            Just <|\n                foldl\n                    (\\current highest ->\n                        if current > highest then\n                            current\n\n                        else\n                            highest\n                    )\n                    val\n                    array\n\n\n-- MODIFY\n\n\n{-| Replaces the element at the given index, or returns the array unmodified if the index is out of bounds.\nPassing a negative index means you want to replace an element counting backwards from the end of the array.\n\n    set 1 10 [ 1, 2, 3 ] == [ 1, 10, 3 ]\n    set 10 10 [ 1, 2, 3 ] == [ 1, 2, 3 ]\n    set -1 10 [ 1, 2, 3 ] == [ 1, 2, 10 ]\n\n-}\nset : Int -> a -> Array a -> Array a\nset =\n    Gren.Kernel.Array.set\n\n\n{-| Update a value at the given index using a function. If the index is out of bounds, nothing happens.\n\n    update 1 (\\n -> n + 1) [ 1, 2, 3 ] == [ 1, 3, 3 ]\n    update 10 (\\n -> n + 1) [ 1, 2, 3 ] == [ 1, 2, 3 ]\n\n-}\nupdate : Int -> (a -> a) -> Array a -> Array a\nupdate idx fn array =\n    case get idx array of\n        Nothing ->\n            array\n\n        Just val ->\n            set idx (fn val) array\n\n\n{-| Add a value to the start of the array.\n\n    pushFirst 1 []          == [ 1 ]\n    pushFirst 5 [ 1, 4, 9 ] == [ 5, 1, 4, 9 ]\n\n-}\npushFirst : a -> Array a -> Array a\npushFirst value array =\n    prepend [ value ] array\n\n\n{-| Add a value to the end of the array.\n\n    pushLast 1 []          == [ 1 ]\n    pushLast 5 [ 1, 4, 9 ] == [ 1, 4, 9, 5 ]\n\n-}\npushLast : a -> Array a -> Array a\npushLast =\n    Gren.Kernel.Array.push\n\n\n-- COMBINE\n\n\n{-| Combine two arrays so that the first array becomes the prefix,\nand the second array becomes the postfix of the resulting array.\n\n    prepend [ 1, 2, 3 ] [ 4, 5, 6 ] == [ 1, 2, 3, 4, 5, 6 ] \n\nYou can also use the `++` operator for this purpose.\n\n    [ 1, 2, 3 ] ++ [ 4, 5, 6 ] == [ 1, 2, 3, 4, 5, 6 ]\n-}\nprepend : Array a -> Array a -> Array a\nprepend =\n    Gren.Kernel.Array.append\n\n\n{-| Combine two arrays so that the first array becomes the postfix,\nand the second array becomes the prefix of the resulting array.\n\n    append [ 1, 2, 3 ] [ 4, 5, 6 ] == [ 4, 5, 6, 1, 2, 3 ]\n    \n-}\nappend : Array a -> Array a -> Array a\nappend fst second =\n    prepend second fst\n\n\n{-| Combine a bunch of arrays into a single array.\n\n    flatten [ [ 1 ], [ 2 ], [ 4, 5 ] ] == [ 1, 2, 4, 5 ]\n\n-}\nflatten : Array (Array a) -> Array a\nflatten =\n    Gren.Kernel.Array.flat\n\n\n{-| Map a given function onto an array, then flatten the resulting array.\n\n    flatMap f xs == flatten (map f xs)\n\n-}\nflatMap : (a -> Array b) -> Array a -> Array b\nflatMap =\n    Gren.Kernel.Array.flatMap\n\n\n{-| Places the given value between all members of the given list.\n\n    intersperse \"on\" [ \"turtles\", \"turtles\", \"turtles\"] == [ \"turtles\", \"on\", \"turtles\", \"on\", \"turtles\"]\n\n-}\nintersperse : a -> Array a -> Array a\nintersperse sep xs =\n    case popFirst xs of\n        Nothing ->\n            []\n\n        Just { first = head, rest = tail } ->\n            pushFirst head <| flatMap (\\val -> [ sep, val ]) tail\n\n\n{-| Combine two arrays, combining them with the given function.\nIf one array is longer, the extra elements are dropped.\n\n    map2 (\\x y -> { x = x, y = y }) [ 1 ] [ 2 ] == [ { x = 1, y = 2 } ]\n-}\nmap2 : (a -> b -> result) -> Array a -> Array b -> Array result\nmap2 =\n    Gren.Kernel.Array.map2\n\n\n{-| Combine three arrays, combining them with the given function.\nIf one array is longer, the extra elements are dropped.\n\n    map3 (\\x y z -> { x = x, y = y, z = z }) [ 1 ] [ 2 ] [ 3 ] == [ { x = 1, y = 2, z = 3 } ]\n-}\nmap3 : (a -> b -> c -> result) -> Array a -> Array b -> Array c -> Array result\nmap3 =\n    Gren.Kernel.Array.map3\n\n\n-- DECONSTRUCT\n\n\n{-| Retrieve the first element of the array, if it exists.\n\n    first [ 1, 2, 3 ] == Just 1\n\n-}\nfirst : Array a -> Maybe a\nfirst array =\n    get 0 array\n\n\n{-| Retrieve the last element of the array, if it exists.\n\n    last [ 1, 2, 3 ] == Just 3\n\n-}\nlast : Array a -> Maybe a\nlast array =\n    get -1 array\n\n\n{-| Get a sub section of an array: `(slice start end array)`.\n\nThe `start` is a zero-based index where we will start our slice.\nThe `end` is a zero-based index that indicates the end of the slice.\nThe slice extracts up to, but no including, the `end`.\n\nBoth `start` and `end` can be negative, indicating an offset from the end\nof the array. Removing the last element of the array can be expressed as:\n\n    `slice 0 -1 arr`.\n\nIn the case of an impossible slice, the empty array is returned.\n\n-}\nslice : Int -> Int -> Array a -> Array a\nslice =\n    Gren.Kernel.Array.slice\n\n\n{-| Remove the first `n` elements of the array.\n\n    dropFirst 5 [ 1 ] == []\n    dropFirst 1 [ 1, 2, 3 ] == [ 2, 3 ]\n\n-}\ndropFirst : Int -> Array a -> Array a\ndropFirst n array =\n    slice n (length array) array\n\n\n{-| Remove the last `n` elements of the array.\n\n    dropLast 1 [ 1, 2, 3 ] == [ 1, 2 ]\n\n-}\ndropLast : Int -> Array a -> Array a\ndropLast n array =\n    slice 0 (length array - n) array\n\n\n{-| Take the first `n` elements from the array.\n\n    takeFirst 2 [ 1, 2, 3 ] == [ 1, 2 ]\n\n-}\ntakeFirst : Int -> Array a -> Array a\ntakeFirst n array =\n    slice 0 n array\n\n\n{-| Take the last `n` elements from the array.\n\n    takeLast 2 [ 1, 2, 3 ] == [ 2, 3 ]\n\n-}\ntakeLast : Int -> Array a -> Array a\ntakeLast n array =\n    let\n        len =\n            length array\n    in\n    slice (len - n) len array\n\n\n{-| Split an array into its first element, and its remaining elements, if possible.\n\n    popFirst [ 1, 2, 3 ] == Just { first = 1, rest = [ 2, 3 ] }\n\n-}\npopFirst : Array a -> Maybe { first : a, rest : Array a }\npopFirst array =\n    case first array of\n        Just value ->\n            Just\n                { first = value\n                , rest = dropFirst 1 array\n                }\n\n        Nothing ->\n            Nothing\n\n\n{-| Split an array into its last element, and its remaining elements, if possible.\n\n    popFirst [ 1, 2, 3 ] == Just { last = 3, initial = [ 1, 2 ] }\n\n-}\npopLast : Array a -> Maybe { last : a, initial : Array a }\npopLast array =\n    case last array of\n        Just value ->\n            Just\n                { last = value\n                , initial = dropLast 1 array\n                }\n\n        Nothing ->\n            Nothing\n\n\n{-| Divide elements into two arrays based on the result of a boolean test.\n\n    partition (\\x -> x < 3) [ 0, 1, 2, 3, 4, 5 ] == { trues = [ 0, 1, 2 ], falses = [ 3, 4, 5 ] }\n\n-}\npartition : (a -> Bool) -> Array a -> { trues : Array a, falses : Array a }\npartition fn array =\n    foldl\n        (\\val { trues, falses } ->\n            if fn val then\n                { trues = pushLast val trues\n                , falses = falses\n                }\n\n            else\n                { trues = trues\n                , falses = pushLast val falses\n                }\n        )\n        { trues = empty, falses = empty }\n        array\n\n\n-- SORT\n\n\n{-| Sort values from lowest to highest\n\n    sort [ 3, 1, 5 ] == [ 1, 3, 5 ]\n\n-}\nsort : Array comparable -> Array comparable\nsort =\n    Gren.Kernel.Array.sort\n\n\n{-| Sort values by a derived property.\n\n    sortBy String.length [ \"mouse\", \"cat\" ] == [ \"cat\", \"mouse\" ]\n\n-}\nsortBy : (a -> comparable) -> Array a -> Array a\nsortBy =\n    Gren.Kernel.Array.sortBy\n\n\n{-| Sort values with a custom comparison function.\n\n    sortWith flippedComparison [1,2,3,4,5] == [5,4,3,2,1]\n\n    flippedComparison a b =\n        case compare a b of\n          LT -> GT\n          EQ -> EQ\n          GT -> LT\n\nThis is also the most general sort function, allowing you to define any other: `sort == sortWith compare`\n\n-}\nsortWith : (a -> a -> Order) -> Array a -> Array a\nsortWith =\n    Gren.Kernel.Array.sortWith\n\n",
        "module Set exposing\n    ( Set\n    , empty, singleton, set, remove\n    , isEmpty, member, count, first, last, findFirst, findLast, any, all\n    , union, intersect, diff\n    , toArray, fromArray\n    , map, foldl, foldr, filter, filterMap, partition\n    )\n\n{-| A set of unique values. The values can be any comparable type. This\nincludes `Int`, `Float`, `Time`, `Char`, `String`, and tuples or lists\nof comparable types.\n\nSet, remove, and query operations all take _O(log n)_ time.\n\n\n@docs Set\n\n\n@docs empty, singleton, set, remove\n\n\n## Query\n\n@docs isEmpty, member, count, first, last, findFirst, findLast, any, all\n\n\n## Combine\n\n@docs union, intersect, diff\n\n\n## Arrays\n\n@docs toArray, fromArray\n\n\n## Transform\n\n@docs map, foldl, foldr, filter, filterMap, partition\n\n-}\n\nimport Array exposing (Array)\nimport Basics exposing (Bool, Int)\nimport Dict\nimport Maybe exposing (Maybe(..))\n\n\n{-| Represents a set of unique values. So `(Set Int)` is a set of integers and\n`(Set String)` is a set of strings.\n-}\ntype Set t\n    = Set_gren_builtin (Dict.Dict t {})\n\n\n{-| Create an empty set.\n-}\nempty : Set a\nempty =\n    Set_gren_builtin Dict.empty\n\n\n{-| Create a set with one value.\n-}\nsingleton : comparable -> Set comparable\nsingleton key =\n    Set_gren_builtin (Dict.singleton key {})\n\n\n{-| Set a value into a set.\n-}\nset : comparable -> Set comparable -> Set comparable\nset key (Set_gren_builtin dict) =\n    Set_gren_builtin (Dict.set key {} dict)\n\n\n{-| Remove a value from a set. If the value is not found, no changes are made.\n-}\nremove : comparable -> Set comparable -> Set comparable\nremove key (Set_gren_builtin dict) =\n    Set_gren_builtin (Dict.remove key dict)\n\n\n{-| Determine if a set is empty.\n-}\nisEmpty : Set a -> Bool\nisEmpty (Set_gren_builtin dict) =\n    Dict.isEmpty dict\n\n\n{-| Determine if a value is in a set.\n-}\nmember : comparable -> Set comparable -> Bool\nmember key (Set_gren_builtin dict) =\n    Dict.member key dict\n\n\n{-| Determine the number of elements in a set.\n-}\ncount : Set a -> Int\ncount (Set_gren_builtin dict) =\n    Dict.count dict\n\n\n{-| Get the first element of the set.\n-}\nfirst : Set a -> Maybe a\nfirst (Set_gren_builtin dict) =\n    Maybe.map .key (Dict.first dict)\n\n\n{-| Get the last element of the set.\n-}\nlast : Set a -> Maybe a\nlast (Set_gren_builtin dict) =\n    Maybe.map .key (Dict.last dict)\n\n\n{-| Find the first value that passes the test.\n-}\nfindFirst : (a -> Bool) -> Set a -> Maybe a\nfindFirst fn (Set_gren_builtin dict) =\n    Maybe.map .key (Dict.findFirst (\\key _ -> fn key) dict)\n\n\n{-| Find the last value that passes the test.\n-}\nfindLast : (a -> Bool) -> Set a -> Maybe a\nfindLast fn (Set_gren_builtin dict) =\n    Maybe.map .key (Dict.findLast (\\key _ -> fn key) dict)\n\n\n{-| Checks if any value in the set passes the test.\n-}\nany : (a -> Bool) -> Set a -> Bool\nany fn (Set_gren_builtin dict) =\n    Dict.any (\\key _ -> fn key) dict\n\n\n{-| Checks if all values in the set passes the test.\n-}\nall : (a -> Bool) -> Set a -> Bool\nall fn (Set_gren_builtin dict) =\n    Dict.all (\\key _ -> fn key) dict\n\n\n{-| Get the union of two sets. Keep all values.\n-}\nunion : Set comparable -> Set comparable -> Set comparable\nunion (Set_gren_builtin dict1) (Set_gren_builtin dict2) =\n    Set_gren_builtin (Dict.union dict1 dict2)\n\n\n{-| Get the intersection of two sets. Keeps values that appear in both sets.\n-}\nintersect : Set comparable -> Set comparable -> Set comparable\nintersect (Set_gren_builtin dict1) (Set_gren_builtin dict2) =\n    Set_gren_builtin (Dict.intersect dict1 dict2)\n\n\n{-| Get the difference between the first set and the second. Keeps values\nthat do not appear in the second set.\n-}\ndiff : Set comparable -> Set comparable -> Set comparable\ndiff (Set_gren_builtin dict1) (Set_gren_builtin dict2) =\n    Set_gren_builtin (Dict.diff dict1 dict2)\n\n\n{-| Convert a set into a list, sorted from lowest to highest.\n-}\ntoArray : Set a -> Array a\ntoArray (Set_gren_builtin dict) =\n    Dict.keys dict\n\n\n{-| Convert a list into a set, removing any duplicates.\n-}\nfromArray : Array comparable -> Set comparable\nfromArray list =\n    Array.foldl set empty list\n\n\n{-| Fold over the values in a set, in order from lowest to highest.\n-}\nfoldl : (a -> b -> b) -> b -> Set a -> b\nfoldl func initialState (Set_gren_builtin dict) =\n    Dict.foldl (\\key _ state -> func key state) initialState dict\n\n\n{-| Fold over the values in a set, in order from highest to lowest.\n-}\nfoldr : (a -> b -> b) -> b -> Set a -> b\nfoldr func initialState (Set_gren_builtin dict) =\n    Dict.foldr (\\key _ state -> func key state) initialState dict\n\n\n{-| Map a function onto a set, creating a new set with no duplicates.\n-}\nmap : (comparable -> comparable2) -> Set comparable -> Set comparable2\nmap func coll =\n    foldl (\\x xs -> set (func x) xs) empty coll\n\n\n{-| Only keep elements that pass the given test.\n\n    import Set exposing (Set)\n\n    numbers : Set Int\n    numbers =\n        Set.fromArray [ -2, -1, 0, 1, 2 ]\n\n    positives : Set Int\n    positives =\n        Set.filter (\\x -> x > 0) numbers\n\n    -- positives == Set.fromArray [1,2]\n\n-}\nfilter : (comparable -> Bool) -> Set comparable -> Set comparable\nfilter isGood (Set_gren_builtin dict) =\n    Set_gren_builtin (Dict.filter (\\key _ -> isGood key) dict)\n\n\n{-| Filter out unwanted results of a map operation.\n    \n    import Set\n\n    strings : Set String\n    strings =\n        Set.fromArray [ \"3\", \"not a number\", \"-5\" ]\n\n    numbers : Set Int\n    numbers =\n        Set.filterMap String.toInt strings\n\n    -- numbers == Set.fromArray [ 3, -5 ]\n-}\nfilterMap : (comparable -> Maybe comparable2) -> Set comparable -> Set comparable2\nfilterMap toMaybe coll =\n    foldl\n        (\\old news ->\n            case toMaybe old of\n                Just new ->\n                    set new news\n\n                Nothing ->\n                    news\n        )\n        empty\n        coll\n\n\n{-| Create two new sets. The first contains all the elements that passed the\ngiven test, and the second contains all the elements that did not.\n-}\npartition : (comparable -> Bool) -> Set comparable -> { trues : Set comparable, falses : Set comparable }\npartition isGood (Set_gren_builtin dict) =\n    let\n        { trues, falses } =\n            Dict.partition (\\key _ -> isGood key) dict\n    in\n    { trues = Set_gren_builtin trues\n    , falses = Set_gren_builtin falses\n    }\n",
        "module String exposing\n    ( String, isEmpty, length, reverse, repeat, replace\n    , prepend, append, split, join, words, lines\n    , slice, left, right, dropLeft, dropRight\n    , contains, startsWith, endsWith, indices\n    , toInt, fromInt\n    , toFloat, fromFloat\n    , fromChar, cons, uncons\n    , toArray, fromArray\n    , toUpper, toLower, pad, padLeft, padRight, trim, trimLeft, trimRight\n    , map, filter, foldl, foldr, any, all\n    )\n\n{-| A built-in representation for efficient string manipulation. String literals\nare enclosed in `\"double quotes\"`.\n\n@docs String, isEmpty, length, reverse, repeat, replace\n\n\n## Building and Splitting\n\n@docs prepend, append, split, join, words, lines\n\n\n## Get Substrings\n\n@docs slice, left, right, dropLeft, dropRight\n\n\n## Check for Substrings\n\n@docs contains, startsWith, endsWith, indices\n\n\n## Int Conversions\n\n@docs toInt, fromInt\n\n\n## Float Conversions\n\n@docs toFloat, fromFloat\n\n\n## Char Conversions\n\n@docs fromChar, cons, uncons\n\n\n## Array Conversions\n\n@docs toArray, fromArray\n\n\n## Formatting\n\nCosmetic operations such as padding with extra characters or trimming whitespace.\n\n@docs toUpper, toLower, pad, padLeft, padRight, trim, trimLeft, trimRight\n\n## Higher-Order Functions\n\n@docs map, filter, foldl, foldr, any, all\n\n-}\n\nimport Array exposing (Array)\nimport Basics exposing (..)\nimport Math exposing (floor, ceiling)\nimport Bitwise\nimport Char exposing (Char)\nimport Gren.Kernel.String\nimport Maybe exposing (Maybe)\nimport Result exposing (Result)\n\n\n\n-- STRINGS\n\n\n{-| A `String` is a chunk of text:\n\n    \"Hello!\"\n\n    \"How are you?\"\n\n    \"🙈🙉🙊\"\n\n    -- strings with escape characters\n    \"this\\n\\t\\\"that\\\"\"\n\n    \"🙈🙉🙊\" -- \"🙈🙉🙊\"\n\n    -- multiline strings\n    \"\"\"Triple double quotes let you\n    create \"multiline strings\" which\n    can have unescaped quotes and newlines.\n    \"\"\"\n\nA `String` can represent any sequence of [unicode characters][u]. You can use\nthe unicode escapes from `\\u{0000}` to `\\u{10FFFF}` to represent characters\nby their code point. You can also include the unicode characters directly.\nUsing the escapes can be better if you need one of the many whitespace\ncharacters with different widths.\n\n[u]: https://en.wikipedia.org/wiki/Unicode\n\n**Note:** JavaScript lets you use double quotes and single quotes interchangably.\nThis is not true in Gren. You must use double quotes for a `String`, and you must\nuse single quotes for a [`Char`](Char#Char).\n\n-}\ntype String\n    = String -- NOTE: The compiler provides the real implementation.\n\n\n{-| Determine if a string is empty.\n\n    isEmpty \"\" == True\n\n    isEmpty \"the world\" == False\n\n-}\nisEmpty : String -> Bool\nisEmpty string =\n    string == \"\"\n\n\n{-| Get the length of a string.\n\n    length \"innumerable\" == 11\n\n    length \"\" == 0\n\n-}\nlength : String -> Int\nlength =\n    Gren.Kernel.String.length\n\n\n{-| Reverse a string.\n\n    reverse \"stressed\" == \"desserts\"\n\n-}\nreverse : String -> String\nreverse =\n    Gren.Kernel.String.reverse\n\n\n{-| Repeat a string _n_ times.\n\n    repeat 3 \"ha\" == \"hahaha\"\n\n-}\nrepeat : Int -> String -> String\nrepeat n chunk =\n    repeatHelp n chunk \"\"\n\n\nrepeatHelp : Int -> String -> String -> String\nrepeatHelp n chunk result =\n    if n <= 0 then\n        result\n\n    else\n        repeatHelp (Bitwise.shiftRightBy 1 n) (chunk ++ chunk) <|\n            if Bitwise.and n 1 == 0 then\n                result\n\n            else\n                result ++ chunk\n\n\n{-| Replace all occurrences of some substring.\n\n    replace \".\" \"-\" \"Json.Decode.succeed\" == \"Json-Decode-succeed\"\n\n    replace \",\" \"/\" \"a,b,c,d,e\" == \"a/b/c/d/e\"\n\n**Note:** If you need more advanced replacements, check out the\n[`gren-lang/parser`][parser] or [`String.Regex`][regex] package.\n\n[parser]: /package/gren-lang/parser\n[regex]: String.Regex\n\n-}\nreplace : String -> String -> String -> String\nreplace before after string =\n    join after (split before string)\n\n\n\n-- BUILDING AND SPLITTING\n\n\n{-| Combine two strings. You can also use [the `(++)` operator](Basics#++)\nto do this.\n\n    prepend \"butter\" \"fly\" == \"butterfly\"\n\n-}\nprepend : String -> String -> String\nprepend =\n    Gren.Kernel.String.append\n\n\n{-| Append one string onto another. This is the same operation as [prepend](prepend),\nbut with the arguments reversed.\n    \n    append \"butter\" \"fly\" == \"flybutter\"\n\n-}\nappend : String -> String -> String\nappend lhs rhs =\n    prepend rhs lhs\n\n\n{-| Split a string using a given separator.\n\n    split \",\" \"cat,dog,cow\" == [ \"cat\", \"dog\", \"cow\" ]\n\n    split \"/\" \"home/evan/Desktop/\" == [ \"home\", \"evan\", \"Desktop\", \"\" ]\n\n-}\nsplit : String -> String -> Array String\nsplit =\n    Gren.Kernel.String.split\n\n\n{-| Put many strings together with a given separator.\n\n    join \"a\" [ \"H\", \"w\", \"ii\", \"n\" ] == \"Hawaiian\"\n\n    join \" \" [ \"cat\", \"dog\", \"cow\" ] == \"cat dog cow\"\n\n    join \"/\" [ \"home\", \"evan\", \"Desktop\" ] == \"home/evan/Desktop\"\n\n-}\njoin : String -> Array String -> String\njoin =\n    Gren.Kernel.String.join\n\n\n{-| Break a string into words, splitting on chunks of whitespace.\n\n    words \"How are \\t you? \\n Good?\" == [ \"How\", \"are\", \"you?\", \"Good?\" ]\n\n-}\nwords : String -> Array String\nwords =\n    Gren.Kernel.String.words\n\n\n{-| Break a string into lines, splitting on newlines.\n\n    lines \"How are you?\\nGood?\" == [ \"How are you?\", \"Good?\" ]\n\n-}\nlines : String -> Array String\nlines =\n    Gren.Kernel.String.lines\n\n\n\n-- SUBSTRINGS\n\n\n{-| Take a substring given a start and end index. Negative indexes\nare taken starting from the _end_ of the list.\n\n    slice 7 9 \"snakes on a plane!\" == \"on\"\n\n    slice 0 6 \"snakes on a plane!\" == \"snakes\"\n\n    slice 0 -7 \"snakes on a plane!\" == \"snakes on a\"\n\n    slice -6 -1 \"snakes on a plane!\" == \"plane\"\n\n-}\nslice : Int -> Int -> String -> String\nslice =\n    Gren.Kernel.String.slice\n\n\n{-| Take _n_ characters from the left side of a string.\n\n    left 2 \"Mulder\" == \"Mu\"\n\n-}\nleft : Int -> String -> String\nleft n string =\n    if n < 1 then\n        \"\"\n\n    else\n        slice 0 n string\n\n\n{-| Take _n_ characters from the right side of a string.\n\n    right 2 \"Scully\" == \"ly\"\n\n-}\nright : Int -> String -> String\nright n string =\n    if n < 1 then\n        \"\"\n\n    else\n        slice -n (length string) string\n\n\n{-| Drop _n_ characters from the left side of a string.\n\n    dropLeft 2 \"The Lone Gunmen\" == \"e Lone Gunmen\"\n\n-}\ndropLeft : Int -> String -> String\ndropLeft n string =\n    if n < 1 then\n        string\n\n    else\n        slice n (length string) string\n\n\n{-| Drop _n_ characters from the right side of a string.\n\n    dropRight 2 \"Cigarette Smoking Man\" == \"Cigarette Smoking M\"\n\n-}\ndropRight : Int -> String -> String\ndropRight n string =\n    if n < 1 then\n        string\n\n    else\n        slice 0 -n string\n\n\n\n-- DETECT SUBSTRINGS\n\n\n{-| See if the second string contains the first one.\n\n    contains \"the\" \"theory\" == True\n\n    contains \"hat\" \"theory\" == False\n\n    contains \"THE\" \"theory\" == False\n\n-}\ncontains : String -> String -> Bool\ncontains =\n    Gren.Kernel.String.contains\n\n\n{-| See if the second string starts with the first one.\n\n    startsWith \"the\" \"theory\" == True\n\n    startsWith \"ory\" \"theory\" == False\n\n-}\nstartsWith : String -> String -> Bool\nstartsWith =\n    Gren.Kernel.String.startsWith\n\n\n{-| See if the second string ends with the first one.\n\n    endsWith \"the\" \"theory\" == False\n\n    endsWith \"ory\" \"theory\" == True\n\n-}\nendsWith : String -> String -> Bool\nendsWith =\n    Gren.Kernel.String.endsWith\n\n\n{-| Get all of the indices for a substring in another string.\n\n    indexes \"i\" \"Mississippi\" == [ 1, 4, 7, 10 ]\n\n    indexes \"ss\" \"Mississippi\" == [ 2, 5 ]\n\n    indexes \"needle\" \"haystack\" == []\n\n-}\nindices : String -> String -> Array Int\nindices =\n    Gren.Kernel.String.indexes\n\n\n\n-- FORMATTING\n\n\n{-| Convert a string to all upper case. Useful for case-insensitive comparisons\nand VIRTUAL YELLING.\n\n    toUpper \"skinner\" == \"SKINNER\"\n\n-}\ntoUpper : String -> String\ntoUpper =\n    Gren.Kernel.String.toUpper\n\n\n{-| Convert a string to all lower case. Useful for case-insensitive comparisons.\n\n    toLower \"X-FILES\" == \"x-files\"\n\n-}\ntoLower : String -> String\ntoLower =\n    Gren.Kernel.String.toLower\n\n\n{-| Pad a string on both sides until it has a given length.\n\n    pad 5 ' ' \"1\" == \"  1  \"\n\n    pad 5 ' ' \"11\" == \"  11 \"\n\n    pad 5 ' ' \"121\" == \" 121 \"\n\n-}\npad : Int -> Char -> String -> String\npad n char string =\n    let\n        half =\n            Basics.toFloat (n - length string) / 2\n    in\n    repeat (ceiling half) (fromChar char) ++ string ++ repeat (floor half) (fromChar char)\n\n\n{-| Pad a string on the left until it has a given length.\n\n    padLeft 5 '.' \"1\" == \"....1\"\n\n    padLeft 5 '.' \"11\" == \"...11\"\n\n    padLeft 5 '.' \"121\" == \"..121\"\n\n-}\npadLeft : Int -> Char -> String -> String\npadLeft n char string =\n    repeat (n - length string) (fromChar char) ++ string\n\n\n{-| Pad a string on the right until it has a given length.\n\n    padRight 5 '.' \"1\" == \"1....\"\n\n    padRight 5 '.' \"11\" == \"11...\"\n\n    padRight 5 '.' \"121\" == \"121..\"\n\n-}\npadRight : Int -> Char -> String -> String\npadRight n char string =\n    string ++ repeat (n - length string) (fromChar char)\n\n\n{-| Get rid of whitespace on both sides of a string.\n\n    trim \"  hats  \\n\" == \"hats\"\n\n-}\ntrim : String -> String\ntrim =\n    Gren.Kernel.String.trim\n\n\n{-| Get rid of whitespace on the left of a string.\n\n    trimLeft \"  hats  \\n\" == \"hats  \\n\"\n\n-}\ntrimLeft : String -> String\ntrimLeft =\n    Gren.Kernel.String.trimLeft\n\n\n{-| Get rid of whitespace on the right of a string.\n\n    trimRight \"  hats  \\n\" == \"  hats\"\n\n-}\ntrimRight : String -> String\ntrimRight =\n    Gren.Kernel.String.trimRight\n\n\n\n-- INT CONVERSIONS\n\n\n{-| Try to convert a string into an int, failing on improperly formatted strings.\n\n    String.toInt \"123\" == Just 123\n\n    String.toInt \"-42\" == Just -42\n\n    String.toInt \"3.1\" == Nothing\n\n    String.toInt \"31a\" == Nothing\n\nIf you are extracting a number from some raw user input, you will typically\nwant to use [`Maybe.withDefault`](Maybe#withDefault) to handle bad data:\n\n    Maybe.withDefault 0 (String.toInt \"42\") == 42\n\n    Maybe.withDefault 0 (String.toInt \"ab\") == 0\n\n-}\ntoInt : String -> Maybe Int\ntoInt =\n    Gren.Kernel.String.toInt\n\n\n{-| Convert an `Int` to a `String`.\n\n    String.fromInt 123 == \"123\"\n\n    String.fromInt -42 == \"-42\"\n\nCheck out [`Debug.toString`](Debug#toString) to convert _any_ value to a string\nfor debugging purposes.\n\n-}\nfromInt : Int -> String\nfromInt =\n    Gren.Kernel.String.fromNumber\n\n\n\n-- FLOAT CONVERSIONS\n\n\n{-| Try to convert a string into a float, failing on improperly formatted strings.\n\n    String.toFloat \"123\" == Just 123.0\n\n    String.toFloat \"-42\" == Just -42.0\n\n    String.toFloat \"3.1\" == Just 3.1\n\n    String.toFloat \"31a\" == Nothing\n\nIf you are extracting a number from some raw user input, you will typically\nwant to use [`Maybe.withDefault`](Maybe#withDefault) to handle bad data:\n\n    Maybe.withDefault 0 (String.toFloat \"42.5\") == 42.5\n\n    Maybe.withDefault 0 (String.toFloat \"cats\") == 0\n\n-}\ntoFloat : String -> Maybe Float\ntoFloat =\n    Gren.Kernel.String.toFloat\n\n\n{-| Convert a `Float` to a `String`.\n\n    String.fromFloat 123 == \"123\"\n\n    String.fromFloat -42 == \"-42\"\n\n    String.fromFloat 3.9 == \"3.9\"\n\nCheck out [`Debug.toString`](Debug#toString) to convert _any_ value to a string\nfor debugging purposes.\n\n-}\nfromFloat : Float -> String\nfromFloat =\n    Gren.Kernel.String.fromNumber\n\n\n\n-- LIST CONVERSIONS\n\n\n{-| Convert a string to a list of characters.\n\n    toArray \"abc\" == [ 'a', 'b', 'c' ]\n\n    toArray \"🙈🙉🙊\" == [ '🙈', '🙉', '🙊' ]\n\n-}\ntoArray : String -> Array Char\ntoArray string =\n    foldl Array.pushLast [] string\n\n\n{-| Convert a list of characters into a String. Can be useful if you\nwant to create a string primarily by consing, perhaps for decoding\nsomething.\n\n    fromArray [ 'a', 'b', 'c' ] == \"abc\"\n\n    fromArray [ '🙈', '🙉', '🙊' ] == \"🙈🙉🙊\"\n\n-}\nfromArray : Array Char -> String\nfromArray =\n    Gren.Kernel.String.fromArray\n\n\n\n-- CHAR CONVERSIONS\n\n\n{-| Create a string from a given character.\n\n    fromChar 'a' == \"a\"\n\n-}\nfromChar : Char -> String\nfromChar char =\n    cons char \"\"\n\n\n{-| Add a character to the beginning of a string.\n\n    cons 'T' \"he truth is out there\" == \"The truth is out there\"\n\n-}\ncons : Char -> String -> String\ncons =\n    Gren.Kernel.String.cons\n\n\n{-| Split a non-empty string into its head and tail. This lets you\npattern match on strings exactly as you would with lists.\n\n    uncons \"abc\" == Just ( 'a', \"bc\" )\n\n    uncons \"\" == Nothing\n\n-}\nuncons : String -> Maybe { first : Char, rest : String }\nuncons =\n    Gren.Kernel.String.uncons\n\n\n\n-- HIGHER-ORDER FUNCTIONS\n\n\n{-| Transform every character in a string\n\n    map\n        (\\c ->\n            if c == '/' then\n                '.'\n\n            else\n                c\n        )\n        \"a/b/c\"\n        == \"a.b.c\"\n\n-}\nmap : (Char -> Char) -> String -> String\nmap =\n    Gren.Kernel.String.map\n\n\n{-| Keep only the characters that pass the test.\n\n    filter isDigit \"R2-D2\" == \"22\"\n\n-}\nfilter : (Char -> Bool) -> String -> String\nfilter =\n    Gren.Kernel.String.filter\n\n\n{-| Reduce a string from the left.\n\n    foldl cons \"\" \"time\" == \"emit\"\n\n-}\nfoldl : (Char -> b -> b) -> b -> String -> b\nfoldl =\n    Gren.Kernel.String.foldl\n\n\n{-| Reduce a string from the right.\n\n    foldr cons \"\" \"time\" == \"time\"\n\n-}\nfoldr : (Char -> b -> b) -> b -> String -> b\nfoldr =\n    Gren.Kernel.String.foldr\n\n\n{-| Determine whether _any_ characters pass the test.\n\n    any isDigit \"90210\" == True\n\n    any isDigit \"R2-D2\" == True\n\n    any isDigit \"heart\" == False\n\n    any isDigit \"\" == False\n\n-}\nany : (Char -> Bool) -> String -> Bool\nany =\n    Gren.Kernel.String.any\n\n\n{-| Determine whether _all_ characters pass the test.\n\n    all isDigit \"90210\" == True\n\n    all isDigit \"R2-D2\" == False\n\n    all isDigit \"heart\" == False\n\n    all isDigit \"\" = True\n\n-}\nall : (Char -> Bool) -> String -> Bool\nall =\n    Gren.Kernel.String.all\n",
        "module Json.Encode exposing\n    ( encode, Value\n    , string, int, float, bool, null\n    , array, set\n    , object, dict\n    )\n\n{-| Functions for turning Gren values into Json values.\n\n\n@docs encode, Value\n\n\n## Primitives\n\n@docs string, int, float, bool, null\n\n\n## Arrays\n\n@docs array, set\n\n\n## Objects\n\n@docs object, dict\n\n-}\n\nimport Basics exposing (..)\nimport Array exposing (Array)\nimport Dict exposing (Dict)\nimport Set exposing (Set)\nimport String exposing (String)\nimport Gren.Kernel.Json\n\n\n\n-- ENCODE\n\n\n{-| Represents a JavaScript value.\n-}\ntype Value\n    = Value\n\n\n{-| Convert a `Value` into a prettified string. The first argument specifies\nthe amount of indentation in the resulting string.\n\n    import Json.Encode as Encode\n\n    tom : Encode.Value\n    tom =\n        Encode.object\n            [ { key = \"name\", value = Encode.string \"Tom\" }\n            , { key = \"age\", value = Encode.int 42 )\n            ]\n\n    compact =\n        Encode.encode 0 tom\n\n    -- {\"name\":\"Tom\",\"age\":42}\n    readable =\n        Encode.encode 4 tom\n\n    -- {\n    --     \"name\": \"Tom\",\n    --     \"age\": 42\n    -- }\n\n-}\nencode : Int -> Value -> String\nencode =\n    Gren.Kernel.Json.encode\n\n\n\n-- PRIMITIVES\n\n\n{-| Turn a `String` into a JSON string.\n\n    import Json.Encode exposing (encode, string)\n\n\n    -- encode 0 (string \"\")      == \"\\\"\\\"\"\n    -- encode 0 (string \"abc\")   == \"\\\"abc\\\"\"\n    -- encode 0 (string \"hello\") == \"\\\"hello\\\"\"\n\n-}\nstring : String -> Value\nstring =\n    Gren.Kernel.Json.wrap\n\n\n{-| Turn an `Int` into a JSON number.\n\n    import Json.Encode exposing (encode, int)\n\n\n    -- encode 0 (int 42) == \"42\"\n    -- encode 0 (int -7) == \"-7\"\n    -- encode 0 (int 0)  == \"0\"\n\n-}\nint : Int -> Value\nint =\n    Gren.Kernel.Json.wrap\n\n\n{-| Turn a `Float` into a JSON number.\n\n    import Json.Encode exposing (encode, float)\n\n\n    -- encode 0 (float 3.14)     == \"3.14\"\n    -- encode 0 (float 1.618)    == \"1.618\"\n    -- encode 0 (float -42)      == \"-42\"\n    -- encode 0 (float NaN)      == \"null\"\n    -- encode 0 (float Infinity) == \"null\"\n\n**Note:** Floating point numbers are defined in the [IEEE 754 standard][ieee]\nwhich is hardcoded into almost all CPUs. This standard allows `Infinity` and\n`NaN`. [The JSON spec][json] does not include these values, so we encode them\nboth as `null`.\n\n[ieee]: https://en.wikipedia.org/wiki/IEEE_754\n[json]: https://www.json.org/\n\n-}\nfloat : Float -> Value\nfloat =\n    Gren.Kernel.Json.wrap\n\n\n{-| Turn a `Bool` into a JSON boolean.\n\n    import Json.Encode exposing (bool, encode)\n\n\n    -- encode 0 (bool True)  == \"true\"\n    -- encode 0 (bool False) == \"false\"\n\n-}\nbool : Bool -> Value\nbool =\n    Gren.Kernel.Json.wrap\n\n\n\n-- NULLS\n\n\n{-| Create a JSON `null` value.\n\n    import Json.Encode exposing (encode, null)\n\n\n    -- encode 0 null == \"null\"\n\n-}\nnull : Value\nnull =\n    Gren.Kernel.Json.encodeNull\n\n\n\n-- ARRAYS\n\n\n{-| Turn a `Array` into a JSON array.\n\n    import Json.Encode as Encode exposing (array, bool, encode, int, string)\n\n\n    -- encode 0 (array int [1,3,4])       == \"[1,3,4]\"\n    -- encode 0 (array bool [True,False]) == \"[true,false]\"\n    -- encode 0 (array string [\"a\",\"b\"])  == \"\"\"[\"a\",\"b\"]\"\"\"\n\n-}\narray : (a -> Value) -> Array a -> Value\narray func entries =\n    Gren.Kernel.Json.wrap\n        (Array.foldl (Gren.Kernel.Json.addEntry func) (Gren.Kernel.Json.emptyArray {}) entries)\n\n\n{-| Turn an `Set` into a JSON array.\n-}\nset : (a -> Value) -> Set a -> Value\nset func entries =\n    Gren.Kernel.Json.wrap\n        (Set.foldl (Gren.Kernel.Json.addEntry func) (Gren.Kernel.Json.emptyArray {}) entries)\n\n\n\n-- OBJECTS\n\n\n{-| Create a JSON object.\n\n    import Json.Encode as Encode\n\n    tom : Encode.Value\n    tom =\n        Encode.object\n            [ { key = \"name\", value = Encode.string \"Tom\" }\n            , { key = \"age\", value = Encode.int 42 }\n            ]\n\n    -- Encode.encode 0 tom == \"\"\"{\"name\":\"Tom\",\"age\":42}\"\"\"\n\n-}\nobject : Array { key : String, value : Value } -> Value\nobject pairs =\n    Gren.Kernel.Json.wrap\n        (Array.foldl\n            (\\{ key, value } obj -> Gren.Kernel.Json.addField key value obj)\n            (Gren.Kernel.Json.emptyObject {})\n            pairs\n        )\n\n\n{-| Turn a `Dict` into a JSON object.\n\n    import Dict exposing (Dict)\n    import Json.Encode as Encode\n\n    people : Dict String Int\n    people =\n        Dict.fromArray [ { key = \"Tom\", value = 42 }, { key = \"Sue\", value = 38 } ]\n\n    -- Encode.encode 0 (Encode.dict identity Encode.int people)\n    --   == \"\"\"{\"Tom\":42,\"Sue\":38}\"\"\"\n\n-}\ndict : (k -> String) -> (v -> Value) -> Dict k v -> Value\ndict toKey toValue dictionary =\n    Gren.Kernel.Json.wrap\n        (Dict.foldl\n            (\\key value obj -> Gren.Kernel.Json.addField (toKey key) (toValue value) obj)\n            (Gren.Kernel.Json.emptyObject {})\n            dictionary\n        )\n",
        "module Json.Decode exposing\n    ( Decoder, string, bool, int, float\n    , nullable, array, dict, keyValuePairs, oneOrMore\n    , field, at, index\n    , maybe, oneOf\n    , decodeString, decodeValue, Value, Error(..), errorToString\n    , map, map2, map3, map4, map5, map6, map7, map8\n    , lazy, value, null, succeed, fail, andThen\n    )\n\n{-| Turn JSON values into Gren values. We've inherited this from Elm. Definitely check out this [intro to\nJSON decoders][guide] to get a feel for how this library works!\n\n[guide]: https://guide.elm-lang.org/effects/json.html\n\n\n@docs Decoder, string, bool, int, float\n\n\n## Data Structures\n\n@docs nullable, array, dict, keyValuePairs, oneOrMore\n\n\n## Object Primitives\n\n@docs field, at, index\n\n\n## Inconsistent Structure\n\n@docs maybe, oneOf\n\n\n## Run Decoders\n\n@docs decodeString, decodeValue, Value, Error, errorToString\n\n\n## Mapping\n\n@docs map, map2, map3, map4, map5, map6, map7, map8\n\n\n## Fancy Decoding\n\n@docs lazy, value, null, succeed, fail, andThen\n\n-}\n\nimport Basics exposing (..)\nimport Array exposing (Array)\nimport Dict exposing (Dict)\nimport Char\nimport String exposing (String)\nimport Maybe exposing (Maybe(..))\nimport Result exposing (Result(..))\nimport Gren.Kernel.Json\nimport Json.Encode\n\n\n\n-- PRIMITIVES\n\n\n{-| A value that knows how to decode JSON values.\n\nThere is a whole section in `guide.elm-lang.org` about decoders, so [check it\nout](https://guide.elm-lang.org/interop/json.html) for a more comprehensive\nintroduction!\n\n-}\ntype Decoder a\n    = Decoder\n\n\n{-| Decode a JSON string into an Gren `String`.\n\n    decodeString string \"true\"              == Err ...\n    decodeString string \"42\"                == Err ...\n    decodeString string \"3.14\"              == Err ...\n    decodeString string \"\\\"hello\\\"\"         == Ok \"hello\"\n    decodeString string \"{ \\\"hello\\\": 42 }\" == Err ...\n\n-}\nstring : Decoder String\nstring =\n    Gren.Kernel.Json.decodeString\n\n\n{-| Decode a JSON boolean into an Gren `Bool`.\n\n    decodeString bool \"true\"              == Ok True\n    decodeString bool \"42\"                == Err ...\n    decodeString bool \"3.14\"              == Err ...\n    decodeString bool \"\\\"hello\\\"\"         == Err ...\n    decodeString bool \"{ \\\"hello\\\": 42 }\" == Err ...\n\n-}\nbool : Decoder Bool\nbool =\n    Gren.Kernel.Json.decodeBool\n\n\n{-| Decode a JSON number into an Gren `Int`.\n\n    decodeString int \"true\"              == Err ...\n    decodeString int \"42\"                == Ok 42\n    decodeString int \"3.14\"              == Err ...\n    decodeString int \"\\\"hello\\\"\"         == Err ...\n    decodeString int \"{ \\\"hello\\\": 42 }\" == Err ...\n\n-}\nint : Decoder Int\nint =\n    Gren.Kernel.Json.decodeInt\n\n\n{-| Decode a JSON number into an Gren `Float`.\n\n    decodeString float \"true\"              == Err ..\n    decodeString float \"42\"                == Ok 42\n    decodeString float \"3.14\"              == Ok 3.14\n    decodeString float \"\\\"hello\\\"\"         == Err ...\n    decodeString float \"{ \\\"hello\\\": 42 }\" == Err ...\n\n-}\nfloat : Decoder Float\nfloat =\n    Gren.Kernel.Json.decodeFloat\n\n\n\n-- DATA STRUCTURES\n\n\n{-| Decode a nullable JSON value into an Gren value.\n\n    decodeString (nullable int) \"13\"    == Ok (Just 13)\n    decodeString (nullable int) \"42\"    == Ok (Just 42)\n    decodeString (nullable int) \"null\"  == Ok Nothing\n    decodeString (nullable int) \"true\"  == Err ..\n\n-}\nnullable : Decoder a -> Decoder (Maybe a)\nnullable decoder =\n    oneOf\n        [ null Nothing\n        , map Just decoder\n        ]\n\n\n{-| Decode a JSON array into an Gren `Array`.\n\n    decodeString (array int) \"[1,2,3]\" == Ok [ 1, 2, 3 ]\n\n    decodeString (array bool) \"[true,false]\" == Ok [ True, False ]\n\n-}\narray : Decoder a -> Decoder (Array a)\narray =\n    Gren.Kernel.Json.decodeArray\n\n\n{-| Decode a JSON object into an Gren `Dict`.\n\n    decodeString (dict int) \"{ \\\"alice\\\": 42, \\\"bob\\\": 99 }\"\n        == Ok (Dict.empty |> Dict.set \"alice\" 42 |> Dict.set \"bob\" 99)\n\nIf you need the keys (like `\"alice\"` and `\"bob\"`) available in the `Dict`\nvalues as well, I recommend using a (private) intermediate data structure like\n`Info` in this example:\n\n    module User exposing ( User, decoder )\n\n    import Dict\n    import Json.Decode exposing (..)\n\n    type alias User =\n        { name : String\n        , height : Float\n        , age : Int\n        }\n\n    makeUser : String -> Float -> Int -> User\n    makeUser name height age =\n        { name = name\n        , height = height\n        , age = age\n        }\n\n    decoder : Decoder (Dict.Dict String User)\n    decoder =\n        map (Dict.map infoToUser) (dict infoDecoder)\n\n    type alias Info =\n        { height : Float\n        , age : Int\n        }\n\n    makeInfo : Float -> Int -> Info\n    makeInfo height age =\n        { height = height\n        , age = age\n        }\n\n    infoDecoder : Decoder Info\n    infoDecoder =\n        map2 makeInfo\n            (field \"height\" float)\n            (field \"age\" int)\n\n    infoToUser : String -> Info -> User\n    infoToUser name { height, age } =\n        makeUser name height age\n\nSo now JSON like `{ \"alice\": { height: 1.6, age: 33 }}` are turned into\ndictionary values like `Dict.singleton \"alice\" (User \"alice\" 1.6 33)` if\nyou need that.\n\n-}\ndict : Decoder a -> Decoder (Dict String a)\ndict decoder =\n    map (\\pairs -> Array.foldl (\\p coll -> Dict.set p.key p.value coll) Dict.empty pairs) (keyValuePairs decoder)\n\n\n{-| Decode a JSON object into an Gren `Array` of pairs.\n\n    decodeString (keyValuePairs int) \"{ \\\"alice\\\": 42, \\\"bob\\\": 99 }\"\n        == Ok [ { key = \"alice\", value = 42 }, { key = \"bob\", value = 99 } ]\n\n-}\nkeyValuePairs : Decoder a -> Decoder (Array { key : String, value : a })\nkeyValuePairs =\n    Gren.Kernel.Json.decodeKeyValuePairs\n\n\n{-| Decode a JSON array that has one or more elements. This comes up if you\nwant to enable drag-and-drop of files into your application. You would pair\nthis function with [`elm/file`]() to write a `dropDecoder` like this:\n\n    import File exposing (File)\n    import Json.Decoder as D\n\n    type Msg\n        = GotFiles File (Array Files)\n\n    inputDecoder : D.Decoder Msg\n    inputDecoder =\n        D.at [ \"dataTransfer\", \"files\" ] (D.oneOrMore GotFiles File.decoder)\n\nThis captures the fact that you can never drag-and-drop zero files.\n\n-}\noneOrMore : (a -> Array a -> value) -> Decoder a -> Decoder value\noneOrMore toValue decoder =\n    array decoder\n        |> andThen (oneOrMoreHelp toValue)\n\n\noneOrMoreHelp : (a -> Array a -> value) -> Array a -> Decoder value\noneOrMoreHelp toValue xs =\n    case Array.get 0 xs of\n        Nothing ->\n            fail \"a ARRAY with at least ONE element\"\n\n        Just y ->\n            succeed (toValue y (Array.slice 1 (Array.length xs) xs))\n\n\n\n-- OBJECT PRIMITIVES\n\n\n{-| Decode a JSON object, requiring a particular field.\n\n    decodeString (field \"x\" int) \"{ \\\"x\\\": 3 }\" == Ok 3\n\n    decodeString (field \"x\" int) \"{ \\\"x\\\": 3, \\\"y\\\": 4 }\" == Ok 3\n\n    decodeString (field \"x\" int) \"{ \\\"x\\\": true }\"\n        == Err\n        ... decodeString (field \"x\" int) \"{ \\\"y\\\": 4 }\"\n        == Err\n        ... decodeString (field \"name\" string) \"{ \\\"name\\\": \\\"tom\\\" }\"\n        == Ok \"tom\"\n\nThe object _can_ have other fields. Lots of them! The only thing this decoder\ncares about is if `x` is present and that the value there is an `Int`.\n\nCheck out [`map2`](#map2) to see how to decode multiple fields!\n\n-}\nfield : String -> Decoder a -> Decoder a\nfield =\n    Gren.Kernel.Json.decodeField\n\n\n{-| Decode a nested JSON object, requiring certain fields.\n\n    json = \"\"\"{ \"person\": { \"name\": \"tom\", \"age\": 42 } }\"\"\"\n\n    decodeString (at [\"person\", \"name\"] string) json  == Ok \"tom\"\n    decodeString (at [\"person\", \"age\" ] int   ) json  == Ok 42\n\nThis is really just a shorthand for saying things like:\n\n    field \"person\" (field \"name\" string) == at [ \"person\", \"name\" ] string\n\n-}\nat : Array String -> Decoder a -> Decoder a\nat fields decoder =\n    Array.foldr field decoder fields\n\n\n{-| Decode a JSON array, requiring a particular index.\n\n    json = \"\"\"[ \"alice\", \"bob\", \"chuck\" ]\"\"\"\n\n    decodeString (index 0 string) json  == Ok \"alice\"\n    decodeString (index 1 string) json  == Ok \"bob\"\n    decodeString (index 2 string) json  == Ok \"chuck\"\n    decodeString (index 3 string) json  == Err ...\n\n-}\nindex : Int -> Decoder a -> Decoder a\nindex =\n    Gren.Kernel.Json.decodeIndex\n\n\n\n-- WEIRD STRUCTURE\n\n\n{-| Helpful for dealing with optional fields. Here are a few slightly different\nexamples:\n\n    json = \"\"\"{ \"name\": \"tom\", \"age\": 42 }\"\"\"\n\n    decodeString (maybe (field \"age\"    int  )) json == Ok (Just 42)\n    decodeString (maybe (field \"name\"   int  )) json == Ok Nothing\n    decodeString (maybe (field \"height\" float)) json == Ok Nothing\n\n    decodeString (field \"age\"    (maybe int  )) json == Ok (Just 42)\n    decodeString (field \"name\"   (maybe int  )) json == Ok Nothing\n    decodeString (field \"height\" (maybe float)) json == Err ...\n\nNotice the last example! It is saying we _must_ have a field named `height` and\nthe content _may_ be a float. There is no `height` field, so the decoder fails.\n\nPoint is, `maybe` will make exactly what it contains conditional. For optional\nfields, this means you probably want it _outside_ a use of `field` or `at`.\n\n-}\nmaybe : Decoder a -> Decoder (Maybe a)\nmaybe decoder =\n    oneOf\n        [ map Just decoder\n        , succeed Nothing\n        ]\n\n\n{-| Try a bunch of different decoders. This can be useful if the JSON may come\nin a couple different formats. For example, say you want to read an array of\nnumbers, but some of them are `null`.\n\n    import String\n\n    badInt : Decoder Int\n    badInt =\n        oneOf [ int, null 0 ]\n\n    -- decodeString (array badInt) \"[1,2,null,4]\" == Ok [1,2,0,4]\n\nWhy would someone generate JSON like this? Questions like this are not good\nfor your health. The point is that you can use `oneOf` to handle situations\nlike this!\n\nYou could also use `oneOf` to help version your data. Try the latest format,\nthen a few older ones that you still support. You could use `andThen` to be\neven more particular if you wanted.\n\n-}\noneOf : Array (Decoder a) -> Decoder a\noneOf =\n    Gren.Kernel.Json.oneOf\n\n\n\n-- MAPPING\n\n\n{-| Transform a decoder. Maybe you just want to know the length of a string:\n\n    import String\n\n    stringLength : Decoder Int\n    stringLength =\n        map String.length string\n\nIt is often helpful to use `map` with `oneOf`, like when defining `nullable`:\n\n    nullable : Decoder a -> Decoder (Maybe a)\n    nullable decoder =\n        oneOf\n            [ null Nothing\n            , map Just decoder\n            ]\n\n-}\nmap : (a -> value) -> Decoder a -> Decoder value\nmap =\n    Gren.Kernel.Json.map1\n\n\n{-| Try two decoders and then combine the result. We can use this to decode\nobjects with many fields:\n\n\n    type alias Point =\n        { x : Float\n        , y : Float\n        }\n\n    makePoint : Float -> Float -> Point\n    makePoint x y =\n        { x = x\n        , y = y\n        }\n\n    point : Decoder Point\n    point =\n        map2 makePoint (field \"x\" float) (field \"y\" float)\n\n    -- decodeString point \"\"\"{ \"x\": 3, \"y\": 4 }\"\"\" == Ok { x = 3, y = 4 }\n\nIt tries each individual decoder and puts the result together with the `Point`\nconstructor.\n\n-}\nmap2 : (a -> b -> value) -> Decoder a -> Decoder b -> Decoder value\nmap2 =\n    Gren.Kernel.Json.map2\n\n\n{-| Try three decoders and then combine the result. We can use this to decode\nobjects with many fields:\n\n\n    type alias Person =\n        { name : String, age : Int, height : Float }\n\n    makePerson : String -> Int -> Float -> Person\n    makePerson name age height =\n        { name = name\n        , age = age\n        , height = height\n        }\n\n    person : Decoder Person\n    person =\n        map3 makePerson\n            (at [ \"name\" ] string)\n            (at [ \"info\", \"age\" ] int)\n            (at [ \"info\", \"height\" ] float)\n\n    -- json = \"\"\"{ \"name\": \"tom\", \"info\": { \"age\": 42, \"height\": 1.8 } }\"\"\"\n    -- decodeString person json == Ok { name = \"tom\", age = 42, height = 1.8 }\n\nLike `map2` it tries each decoder in order and then give the results to the\n`Person` constructor. That can be any function though!\n\n-}\nmap3 : (a -> b -> c -> value) -> Decoder a -> Decoder b -> Decoder c -> Decoder value\nmap3 =\n    Gren.Kernel.Json.map3\n\n\n{-| -}\nmap4 : (a -> b -> c -> d -> value) -> Decoder a -> Decoder b -> Decoder c -> Decoder d -> Decoder value\nmap4 =\n    Gren.Kernel.Json.map4\n\n\n{-| -}\nmap5 : (a -> b -> c -> d -> e -> value) -> Decoder a -> Decoder b -> Decoder c -> Decoder d -> Decoder e -> Decoder value\nmap5 =\n    Gren.Kernel.Json.map5\n\n\n{-| -}\nmap6 : (a -> b -> c -> d -> e -> f -> value) -> Decoder a -> Decoder b -> Decoder c -> Decoder d -> Decoder e -> Decoder f -> Decoder value\nmap6 =\n    Gren.Kernel.Json.map6\n\n\n{-| -}\nmap7 : (a -> b -> c -> d -> e -> f -> g -> value) -> Decoder a -> Decoder b -> Decoder c -> Decoder d -> Decoder e -> Decoder f -> Decoder g -> Decoder value\nmap7 =\n    Gren.Kernel.Json.map7\n\n\n{-| -}\nmap8 : (a -> b -> c -> d -> e -> f -> g -> h -> value) -> Decoder a -> Decoder b -> Decoder c -> Decoder d -> Decoder e -> Decoder f -> Decoder g -> Decoder h -> Decoder value\nmap8 =\n    Gren.Kernel.Json.map8\n\n\n\n-- RUN DECODERS\n\n\n{-| Parse the given string into a JSON value and then run the `Decoder` on it.\nThis will fail if the string is not well-formed JSON or if the `Decoder`\nfails for some reason.\n\n    decodeString int \"4\"     == Ok 4\n    decodeString int \"1 + 2\" == Err ...\n\n-}\ndecodeString : Decoder a -> String -> Result Error a\ndecodeString =\n    Gren.Kernel.Json.runOnString\n\n\n{-| Run a `Decoder` on some JSON `Value`. You can send these JSON values\nthrough ports, so that is probably the main time you would use this function.\n-}\ndecodeValue : Decoder a -> Value -> Result Error a\ndecodeValue =\n    Gren.Kernel.Json.run\n\n\n{-| Represents a JavaScript value.\n-}\ntype alias Value =\n    Json.Encode.Value\n\n\n{-| A structured error describing exactly how the decoder failed. You can use\nthis to create more elaborate visualizations of a decoder problem. For example,\nyou could show the entire JSON object and show the part causing the failure in\nred.\n-}\ntype Error\n    = Field String Error\n    | Index Int Error\n    | OneOf (Array Error)\n    | Failure String Value\n\n\n{-| Convert a decoding error into a `String` that is nice for debugging.\n\nIt produces multiple lines of output, so you may want to peek at it with\nsomething like this:\n\n    import Html\n    import Json.Decode as Decode\n\n    errorToHtml : Decode.Error -> Html.Html msg\n    errorToHtml error =\n        Html.pre [] [ Html.text (Decode.errorToString error) ]\n\n**Note:** It would be cool to do nicer coloring and fancier HTML, but I wanted\nto avoid having an `elm/html` dependency for now. It is totally possible to\ncrawl the `Error` structure and create this separately though!\n\n-}\nerrorToString : Error -> String\nerrorToString error =\n    errorToStringHelp error []\n\n\nerrorToStringHelp : Error -> Array String -> String\nerrorToStringHelp error context =\n    case error of\n        Field f err ->\n            let\n                isSimple =\n                    case String.uncons f of\n                        Nothing ->\n                            False\n\n                        Just { first = char, rest } ->\n                            Char.isAlpha char && String.all Char.isAlphaNum rest\n\n                fieldName =\n                    if isSimple then\n                        \".\" ++ f\n\n                    else\n                        \"['\" ++ f ++ \"']\"\n            in\n            errorToStringHelp err ([ fieldName ] ++ context)\n\n        Index i err ->\n            let\n                indexName =\n                    \"[\" ++ String.fromInt i ++ \"]\"\n            in\n            errorToStringHelp err ([ indexName ] ++ context)\n\n        OneOf errors ->\n            case errors of\n                [] ->\n                    \"Ran into a Json.Decode.oneOf with no possibilities\"\n                        ++ (case context of\n                                [] ->\n                                    \"!\"\n\n                                _ ->\n                                    \" at json\" ++ String.join \"\" context\n                           )\n\n                [ err ] ->\n                    errorToStringHelp err context\n\n                _ ->\n                    let\n                        starter =\n                            case context of\n                                [] ->\n                                    \"Json.Decode.oneOf\"\n\n                                _ ->\n                                    \"The Json.Decode.oneOf at json\" ++ String.join \"\" context\n\n                        introduction =\n                            starter ++ \" failed in the following \" ++ String.fromInt (Array.length errors) ++ \" ways:\"\n                    in\n                    String.join \"\\n\\n\" ([ introduction ] ++ Array.indexedMap errorOneOf errors)\n\n        Failure msg json ->\n            let\n                introduction =\n                    case context of\n                        [] ->\n                            \"Problem with the given value:\\n\\n\"\n\n                        _ ->\n                            \"Problem with the value at json\" ++ String.join \"\" context ++ \":\\n\\n    \"\n            in\n            introduction ++ indent (Json.Encode.encode 4 json) ++ \"\\n\\n\" ++ msg\n\n\nerrorOneOf : Int -> Error -> String\nerrorOneOf i error =\n    \"\\n\\n(\" ++ String.fromInt (i + 1) ++ \") \" ++ indent (errorToString error)\n\n\nindent : String -> String\nindent str =\n    String.join \"\\n    \" (String.split \"\\n\" str)\n\n\n\n-- FANCY PRIMITIVES\n\n\n{-| Ignore the JSON and produce a certain Gren value.\n\n    decodeString (succeed 42) \"true\"    == Ok 42\n    decodeString (succeed 42) \"[1,2,3]\" == Ok 42\n    decodeString (succeed 42) \"hello\"   == Err ... -- this is not a valid JSON string\n\nThis is handy when used with `oneOf` or `andThen`.\n\n-}\nsucceed : a -> Decoder a\nsucceed =\n    Gren.Kernel.Json.succeed\n\n\n{-| Ignore the JSON and make the decoder fail. This is handy when used with\n`oneOf` or `andThen` where you want to give a custom error message in some\ncase.\n\nSee the [`andThen`](#andThen) docs for an example.\n\n-}\nfail : String -> Decoder a\nfail =\n    Gren.Kernel.Json.fail\n\n\n{-| Create decoders that depend on previous results. If you are creating\nversioned data, you might do something like this:\n\n\n    info : Decoder Info\n    info =\n        field \"version\" int\n            |> andThen infoHelp\n\n    infoHelp : Int -> Decoder Info\n    infoHelp version =\n        case version of\n            4 ->\n                infoDecoder4\n\n            3 ->\n                infoDecoder3\n\n            _ ->\n                fail <|\n                    \"Trying to decode info, but version \"\n                        ++ toString version\n                        ++ \" is not supported.\"\n\n    -- infoDecoder4 : Decoder Info\n    -- infoDecoder3 : Decoder Info\n\n-}\nandThen : (a -> Decoder b) -> Decoder a -> Decoder b\nandThen =\n    Gren.Kernel.Json.andThen\n\n\n{-| Sometimes you have JSON with recursive structure, like nested comments.\nYou can use `lazy` to make sure your decoder unrolls lazily.\n\n    type alias Comment =\n        { message : String\n        , responses : Responses\n        }\n\n    makeComment : String -> Responses -> Comment\n    makeComment message responses =\n        { message = message\n        , responses = responses\n        }\n\n    type Responses\n        = Responses (Array Comment)\n\n    comment : Decoder Comment\n    comment =\n        map2 makeComment\n            (field \"message\" string)\n            (field \"responses\" (map Responses (array (lazy (\\_ -> comment)))))\n\nIf we had said `array comment` instead, we would start expanding the value\ninfinitely. What is a `comment`? It is a decoder for objects where the\n`responses` field contains comments. What is a `comment` though? Etc.\n\nBy using `array (lazy (\\_ -> comment))` we make sure the decoder only expands\nto be as deep as the JSON we are given. You can read more about recursive data\nstructures [here].\n\n[here]: https://github.com/elm/compiler/blob/master/hints/recursive-alias.md\n\n-}\nlazy : ({} -> Decoder a) -> Decoder a\nlazy thunk =\n    andThen thunk (succeed {})\n\n\n{-| Do not do anything with a JSON value, just bring it into Gren as a `Value`.\nThis can be useful if you have particularly complex data that you would like to\ndeal with later. Or if you are going to send it out a port and do not care\nabout its structure.\n-}\nvalue : Decoder Value\nvalue =\n    Gren.Kernel.Json.decodeValue\n\n\n{-| Decode a `null` value into some Gren value.\n\n    decodeString (null False) \"null\" == Ok False\n    decodeString (null 42) \"null\"    == Ok 42\n    decodeString (null 42) \"42\"      == Err ..\n    decodeString (null 42) \"false\"   == Err ..\n\nSo if you ever see a `null`, this will return whatever value you specified.\n\n-}\nnull : a -> Decoder a\nnull =\n    Gren.Kernel.Json.decodeNull\n",
        "module Char exposing\n    ( Char\n    , isUpper, isLower, isAlpha, isAlphaNum\n    , isDigit, isOctDigit, isHexDigit\n    , toUpper, toLower, toLocaleUpper, toLocaleLower\n    , toCode, fromCode\n    )\n\n{-| Functions for working with characters. Character literals are enclosed in\n`'a'` pair of single quotes.\n\n\n@docs Char\n\n\n## ASCII Letters\n\n@docs isUpper, isLower, isAlpha, isAlphaNum\n\n\n## Digits\n\n@docs isDigit, isOctDigit, isHexDigit\n\n\n## Conversion\n\n@docs toUpper, toLower, toLocaleUpper, toLocaleLower\n\n\n## Unicode Code Points\n\n@docs toCode, fromCode\n\n-}\n\nimport Basics exposing ((&&), (<=), (>=), (||), Bool, Int)\nimport Gren.Kernel.Char\n\n\n\n-- CHAR\n\n\n{-| A `Char` is a single [unicode][u] character:\n\n    'a'\n\n    '0'\n\n    'Z'\n\n    '?'\n\n    '\"'\n\n    'Σ'\n\n    '🙈'\n\n    '\\t'\n\n    '\"'\n\n    '\\''\n\n    '🙈' -- '🙈'\n\n**Note 1:** You _cannot_ use single quotes around multiple characters like in\nJavaScript. This is how we distinguish [`String`](String#String) and `Char`\nvalues in syntax.\n\n**Note 2:** You can use the unicode escapes from `\\u{0000}` to `\\u{10FFFF}` to\nrepresent characters by their code point. You can also include the unicode\ncharacters directly. Using the escapes can be better if you need one of the\nmany whitespace characters with different widths.\n\n[u]: https://en.wikipedia.org/wiki/Unicode\n\n-}\ntype Char\n    = Char -- NOTE: The compiler provides the real implementation.\n\n\n\n-- CLASSIFICATION\n\n\n{-| Detect upper case ASCII characters.\n\n    isUpper 'A' == True\n\n    isUpper 'B'\n        == True\n        ... isUpper 'Z'\n        == True\n\n    isUpper '0' == False\n\n    isUpper 'a' == False\n\n    isUpper '-' == False\n\n    isUpper 'Σ' == False\n\n-}\nisUpper : Char -> Bool\nisUpper char =\n    let\n        code =\n            toCode char\n    in\n    code <= 0x5A && 0x41 <= code\n\n\n{-| Detect lower case ASCII characters.\n\n    isLower 'a' == True\n\n    isLower 'b'\n        == True\n        ... isLower 'z'\n        == True\n\n    isLower '0' == False\n\n    isLower 'A' == False\n\n    isLower '-' == False\n\n    isLower 'π' == False\n\n-}\nisLower : Char -> Bool\nisLower char =\n    let\n        code =\n            toCode char\n    in\n    0x61 <= code && code <= 0x7A\n\n\n{-| Detect upper case and lower case ASCII characters.\n\n    isAlpha 'a' == True\n\n    isAlpha 'b' == True\n\n    isAlpha 'E' == True\n\n    isAlpha 'Y' == True\n\n    isAlpha '0' == False\n\n    isAlpha '-' == False\n\n    isAlpha 'π' == False\n\n-}\nisAlpha : Char -> Bool\nisAlpha char =\n    isLower char || isUpper char\n\n\n{-| Detect upper case and lower case ASCII characters.\n\n    isAlphaNum 'a' == True\n\n    isAlphaNum 'b' == True\n\n    isAlphaNum 'E' == True\n\n    isAlphaNum 'Y' == True\n\n    isAlphaNum '0' == True\n\n    isAlphaNum '7' == True\n\n    isAlphaNum '-' == False\n\n    isAlphaNum 'π' == False\n\n-}\nisAlphaNum : Char -> Bool\nisAlphaNum char =\n    isLower char || isUpper char || isDigit char\n\n\n{-| Detect digits `0123456789`\n\n    isDigit '0' == True\n\n    isDigit '1'\n        == True\n        ... isDigit '9'\n        == True\n\n    isDigit 'a' == False\n\n    isDigit 'b' == False\n\n    isDigit 'A' == False\n\n-}\nisDigit : Char -> Bool\nisDigit char =\n    let\n        code =\n            toCode char\n    in\n    code <= 0x39 && 0x30 <= code\n\n\n{-| Detect octal digits `01234567`\n\n    isOctDigit '0' == True\n\n    isOctDigit '1'\n        == True\n        ... isOctDigit '7'\n        == True\n\n    isOctDigit '8' == False\n\n    isOctDigit 'a' == False\n\n    isOctDigit 'A' == False\n\n-}\nisOctDigit : Char -> Bool\nisOctDigit char =\n    let\n        code =\n            toCode char\n    in\n    code <= 0x37 && 0x30 <= code\n\n\n{-| Detect hexadecimal digits `0123456789abcdefABCDEF`\n-}\nisHexDigit : Char -> Bool\nisHexDigit char =\n    let\n        code =\n            toCode char\n    in\n    (0x30 <= code && code <= 0x39)\n        || (0x41 <= code && code <= 0x46)\n        || (0x61 <= code && code <= 0x66)\n\n\n\n-- CONVERSIONS\n\n\n{-| Convert to upper case.\n-}\ntoUpper : Char -> Char\ntoUpper =\n    Gren.Kernel.Char.toUpper\n\n\n{-| Convert to lower case.\n-}\ntoLower : Char -> Char\ntoLower =\n    Gren.Kernel.Char.toLower\n\n\n{-| Convert to upper case, according to any locale-specific case mappings.\n-}\ntoLocaleUpper : Char -> Char\ntoLocaleUpper =\n    Gren.Kernel.Char.toLocaleUpper\n\n\n{-| Convert to lower case, according to any locale-specific case mappings.\n-}\ntoLocaleLower : Char -> Char\ntoLocaleLower =\n    Gren.Kernel.Char.toLocaleLower\n\n\n{-| Convert to the corresponding Unicode [code point][cp].\n\n[cp]: https://en.wikipedia.org/wiki/Code_point\n\n    toCode 'A' == 65\n\n    toCode 'B' == 66\n\n    toCode '木' == 0x6728\n\n    toCode '𝌆' == 0x0001D306\n\n    toCode '😃' == 0x0001F603\n\n-}\ntoCode : Char -> Int\ntoCode =\n    Gren.Kernel.Char.toCode\n\n\n{-| Convert a Unicode [code point][cp] to a character.\n\n    fromCode 65 == 'A'\n\n    fromCode 66 == 'B'\n\n    fromCode 0x6728 == '木'\n\n    fromCode 0x0001D306 == '𝌆'\n\n    fromCode 0x0001F603 == '😃'\n\n    fromCode -1 == '�'\n\nThe full range of unicode is from `0` to `0x10FFFF`. With numbers outside that\nrange, you get [the replacement character][fffd].\n\n[cp]: https://en.wikipedia.org/wiki/Code_point\n[fffd]: https://en.wikipedia.org/wiki/Specials_(Unicode_block)#Replacement_character\n\n-}\nfromCode : Int -> Char\nfromCode =\n    Gren.Kernel.Char.fromCode\n",
        "module Result exposing\n    ( Result(..)\n    , map, map2, map3, map4, map5\n    , andThen\n    , withDefault, toMaybe, fromMaybe, mapError\n    )\n\n{-| A `Result` is the result of a computation that may fail. This is a great\nway to manage errors in Gren.\n\n\n@docs Result\n\n\n## Mapping\n\n@docs map, map2, map3, map4, map5\n\n\n## Chaining\n\n@docs andThen\n\n\n## Handling Errors\n\n@docs withDefault, toMaybe, fromMaybe, mapError\n\n-}\n\nimport Basics exposing (Bool(..))\nimport Maybe exposing (Maybe(..))\n\n\n{-| A `Result` is either `Ok` meaning the computation succeeded, or it is an\n`Err` meaning that there was some failure.\n-}\ntype Result error value\n    = Ok value\n    | Err error\n\n\n{-| If the result is `Ok` return the value, but if the result is an `Err` then\nreturn a given default value. The following examples try to parse integers.\n\n    Result.withDefault 0 (Ok 123) == 123\n\n    Result.withDefault 0 (Err \"no\") == 0\n\n-}\nwithDefault : a -> Result x a -> a\nwithDefault def result =\n    case result of\n        Ok a ->\n            a\n\n        Err _ ->\n            def\n\n\n{-| Apply a function to a result. If the result is `Ok`, it will be converted.\nIf the result is an `Err`, the same error value will propagate through.\n\n    map sqrt (Ok 4.0) == Ok 2.0\n\n    map sqrt (Err \"bad input\") == Err \"bad input\"\n\n-}\nmap : (a -> value) -> Result x a -> Result x value\nmap func ra =\n    case ra of\n        Ok a ->\n            Ok (func a)\n\n        Err e ->\n            Err e\n\n\n{-| Apply a function if both results are `Ok`. If not, the first `Err` will\npropagate through.\n\n    map2 max (Ok 42) (Ok 13) == Ok 42\n\n    map2 max (Err \"x\") (Ok 13) == Err \"x\"\n\n    map2 max (Ok 42) (Err \"y\") == Err \"y\"\n\n    map2 max (Err \"x\") (Err \"y\") == Err \"x\"\n\nThis can be useful if you have two computations that may fail, and you want\nto put them together quickly.\n\n-}\nmap2 : (a -> b -> value) -> Result x a -> Result x b -> Result x value\nmap2 func ra rb =\n    case ra of\n        Err x ->\n            Err x\n\n        Ok a ->\n            case rb of\n                Err x ->\n                    Err x\n\n                Ok b ->\n                    Ok (func a b)\n\n\n{-| -}\nmap3 : (a -> b -> c -> value) -> Result x a -> Result x b -> Result x c -> Result x value\nmap3 func ra rb rc =\n    case ra of\n        Err x ->\n            Err x\n\n        Ok a ->\n            case rb of\n                Err x ->\n                    Err x\n\n                Ok b ->\n                    case rc of\n                        Err x ->\n                            Err x\n\n                        Ok c ->\n                            Ok (func a b c)\n\n\n{-| -}\nmap4 : (a -> b -> c -> d -> value) -> Result x a -> Result x b -> Result x c -> Result x d -> Result x value\nmap4 func ra rb rc rd =\n    case ra of\n        Err x ->\n            Err x\n\n        Ok a ->\n            case rb of\n                Err x ->\n                    Err x\n\n                Ok b ->\n                    case rc of\n                        Err x ->\n                            Err x\n\n                        Ok c ->\n                            case rd of\n                                Err x ->\n                                    Err x\n\n                                Ok d ->\n                                    Ok (func a b c d)\n\n\n{-| -}\nmap5 : (a -> b -> c -> d -> e -> value) -> Result x a -> Result x b -> Result x c -> Result x d -> Result x e -> Result x value\nmap5 func ra rb rc rd re =\n    case ra of\n        Err x ->\n            Err x\n\n        Ok a ->\n            case rb of\n                Err x ->\n                    Err x\n\n                Ok b ->\n                    case rc of\n                        Err x ->\n                            Err x\n\n                        Ok c ->\n                            case rd of\n                                Err x ->\n                                    Err x\n\n                                Ok d ->\n                                    case re of\n                                        Err x ->\n                                            Err x\n\n                                        Ok e ->\n                                            Ok (func a b c d e)\n\n\n{-| Chain together a sequence of computations that may fail. It is helpful\nto see its definition:\n\n    andThen : (a -> Result e b) -> Result e a -> Result e b\n    andThen callback result =\n        case result of\n            Ok value ->\n                callback value\n\n            Err msg ->\n                Err msg\n\nThis means we only continue with the callback if things are going well. For\nexample, say you need to use (`toInt : String -> Result String Int`) to parse\na month and make sure it is between 1 and 12:\n\n\n    toValidMonth : Int -> Result String Int\n    toValidMonth month =\n        if month >= 1 && month <= 12 then\n            Ok month\n\n        else\n            Err \"months must be between 1 and 12\"\n\n    toMonth : String -> Result String Int\n    toMonth rawString =\n        toInt rawString\n            |> andThen toValidMonth\n\n    -- toMonth \"4\" == Ok 4\n    -- toMonth \"9\" == Ok 9\n    -- toMonth \"a\" == Err \"cannot parse to an Int\"\n    -- toMonth \"0\" == Err \"months must be between 1 and 12\"\n\nThis allows us to come out of a chain of operations with quite a specific error\nmessage. It is often best to create a custom type that explicitly represents\nthe exact ways your computation may fail. This way it is easy to handle in your\ncode.\n\n-}\nandThen : (a -> Result x b) -> Result x a -> Result x b\nandThen callback result =\n    case result of\n        Ok value ->\n            callback value\n\n        Err msg ->\n            Err msg\n\n\n{-| Transform an `Err` value. For example, say the errors we get have too much\ninformation:\n\n    parseInt : String -> Result ParseError Int\n\n    type alias ParseError =\n        { message : String\n        , code : Int\n        , position : (Int,Int)\n        }\n\n    mapError .message (parseInt \"123\") == Ok 123\n    mapError .message (parseInt \"abc\") == Err \"char 'a' is not a number\"\n\n-}\nmapError : (x -> y) -> Result x a -> Result y a\nmapError f result =\n    case result of\n        Ok v ->\n            Ok v\n\n        Err e ->\n            Err (f e)\n\n\n{-| Convert to a simpler `Maybe` if the actual error message is not needed or\nyou need to interact with some code that primarily uses maybes.\n\n    parseInt : String -> Result ParseError Int\n\n    maybeParseInt : String -> Maybe Int\n    maybeParseInt string =\n        toMaybe (parseInt string)\n\n-}\ntoMaybe : Result x a -> Maybe a\ntoMaybe result =\n    case result of\n        Ok v ->\n            Just v\n\n        Err _ ->\n            Nothing\n\n\n{-| Convert from a simple `Maybe` to interact with some code that primarily\nuses `Results`.\n\n    parseInt : String -> Maybe Int\n\n    resultParseInt : String -> Result String Int\n    resultParseInt string =\n        fromMaybe (\"error parsing string: \" ++ toString string) (parseInt string)\n\n-}\nfromMaybe : x -> Maybe a -> Result x a\nfromMaybe err maybe =\n    case maybe of\n        Just v ->\n            Ok v\n\n        Nothing ->\n            Err err\n\n\n\n-- FOR INTERNAL USE ONLY\n--\n-- Use `case` expressions for this in Gren code!\n\n\nisOk : Result x a -> Bool\nisOk result =\n    case result of\n        Ok _ ->\n            True\n\n        Err _ ->\n            False\n",
        "module Node exposing\n    ( Environment\n    , Platform(..)\n    , CpuArchitecture(..)\n    , getEnvironmentVariables\n    --\n    , SimpleProgram\n    , defineSimpleProgram\n    , endWithCmd\n    --\n    , Program\n    , ProgramConfiguration\n    , defineProgram\n    , startProgram    \n    --\n    , exit\n    , exitWithCode\n    , setExitCode\n    )\n\n\n{-| A NodeJS program is defined like a browser-based Gren program, except that\nthere is more flexibility regarding how it is initialized.\n\nYou can initialize any number of subsystems, like `FileSystem` or `Terminal`, before\ninitializing your own program with the results of those initializations.\n\nAs part of initializing a subsystem, you usually also get access to a value that permits\nyou to contact said subsystem. Be careful what code you give these permissions to.\n\n## Program\n\n@docs Program, ProgramConfiguration, defineProgram, startProgram\n\n## Simple Program\n\n@docs SimpleProgram, defineSimpleProgram, endWithCmd\n\n## Environment information\n\n@docs Environment, Platform, CpuArchitecture, getEnvironmentVariables\n\n## Exit\n\n@docs exit, exitWithCode, setExitCode\n-}\n\n\nimport Dict exposing ( Dict )\nimport Init\nimport Internal.Init\nimport Task exposing ( Task )\nimport Gren.Kernel.Node\nimport Internal.Stream as IStream\nimport FileSystem.Path exposing (Path)\n\n\n-- ENVIRONMENT\n\n\n{-| Contains information about the environment your application was initiated in.\n\n* `platform` and `cpuArchitecture` tells you something about the operating system and machine your application is running on.\n* `args` is an `Array` of the arguments passed to your application.\n* `stdout`, `stderr` and `stdin` are streams you can use to communicate with the outside world. Take a closer look at the `Stream` module for more information.\n\n-}\ntype alias Environment =\n    { platform : Platform\n    , cpuArchitecture : CpuArchitecture\n    , applicationPath : Path\n    , args : Array String\n    , stdout : IStream.Stream\n    , stderr : IStream.Stream\n    , stdin : IStream.Stream\n    }\n\n\ninitializeEnvironment : Task Never Environment\ninitializeEnvironment =\n    Gren.Kernel.Node.init\n        |> Task.map\n                (\\raw ->\n                    { platform = platformFromString raw.platform\n                    , cpuArchitecture = archFromString raw.arch\n                    , applicationPath = raw.applicationPath\n                    , args = raw.args\n                    , stdout = IStream.Stream 0 raw.stdout\n                    , stderr = IStream.Stream 1 raw.stderr\n                    , stdin = IStream.Stream 2 raw.stdin\n                    }\n                )\n\n\n{-| The platform, or operating system, that your application is running on.\n-}\ntype Platform\n    = Win32\n    | Darwin\n    | Linux\n    | FreeBSD\n    | OpenBSD\n    | SunOS\n    | Aix\n    | UnknownPlatform String\n\n\nplatformFromString : String -> Platform\nplatformFromString platform =\n    case String.toLower platform of\n        \"win32\" ->\n            Win32\n\n        \"darwin\" ->\n            Darwin\n\n        \"linux\" ->\n            Linux\n\n        \"freebsd\" ->\n            FreeBSD\n\n        \"openbsd\" ->\n            OpenBSD\n\n        \"sunos\" ->\n            SunOS\n\n        \"aix\" ->\n            Aix\n\n        _ ->\n            UnknownPlatform platform\n\n\n{-| The CPU architecture your application is running on.\n-}\ntype CpuArchitecture\n    = Arm\n    | Arm64\n    | IA32\n    | Mips\n    | Mipsel\n    | PPC\n    | PPC64\n    | S390\n    | S390x\n    | X64\n    | UnknownArchitecture String\n\n\narchFromString : String -> CpuArchitecture\narchFromString arch =\n    case String.toLower arch of\n        \"arm\" ->\n            Arm\n\n        \"arm64\" ->\n            Arm64\n\n        \"ia32\" ->\n            IA32\n\n        \"mips\" ->\n            Mips\n\n        \"mipsel\" ->\n            Mipsel\n\n        \"ppc\" ->\n            PPC\n\n        \"ppc64\" ->\n            PPC64\n\n        \"s390\" ->\n            S390\n\n        \"s390x\" ->\n            S390x\n\n        \"x64\" ->\n            X64\n\n        _ ->\n            UnknownArchitecture arch\n\n\n{-| Get a `Dict` of environment variables.\n-}\ngetEnvironmentVariables : Task x (Dict String String)\ngetEnvironmentVariables =\n    Gren.Kernel.Node.getEnvironmentVariables\n\n\n-- PROGRAMS\n\n\n{-| A program that executes a single task and then exits.\n-}\ntype alias SimpleProgram a =\n    Program {} a\n\n\n{-| The definition of a Gren program that runs on NodeJS.\n-}\ntype alias Program model msg =\n    Platform.Program {} (Model model) (Msg model msg)\n\n\n-- TOP LEVEL PROGRAM\n\n\ntype Model model\n    = Uninitialized\n    | Initialized model\n\n\ntype Msg model msg\n    = InitDone { model : model, command : Cmd msg }\n    | MsgReceived msg\n\n\n{-| The required functions that define a program.\n-}\ntype alias ProgramConfiguration model msg =\n    { init : Environment -> Init.Task { model : model, command : Cmd msg }\n    , update : msg -> model -> { model : model, command : Cmd msg }\n    , subscriptions : model -> Sub msg\n    }\n\n\n{-| Define a program with access to long-lived state and the ability to respond to\nmessages and listen to subscriptions. If you want to define a simple and short-lived\nprogram, chances are you're looking for [defineSimpleProgram](#defineSimpleProgram) instead.\n-}\ndefineProgram : ProgramConfiguration model msg -> Program model msg\ndefineProgram config =\n    Platform.worker\n        { init = init config.init\n        , update = update config.update\n        , subscriptions = subscriptions config.subscriptions\n        }\n\n\n{-| Define a simple program that doesn't require long-lived state or the ability to respond\nto messages or subscriptions. Ideal for simple and short-lived programs.\n-}\ndefineSimpleProgram : (Environment -> Init.Task (Cmd a)) -> SimpleProgram a\ndefineSimpleProgram initTask =\n    Platform.worker\n        { init = \\_ ->\n            { model = Uninitialized\n            , command =\n                initializeEnvironment\n                    |> Task.andThen (\\env -> unwrap <| initTask env)\n                    |> Task.perform \n                        (\\cmd -> \n                            InitDone\n                                { model = {}\n                                , command = cmd\n                                }\n                        )\n            }\n        , update = update (\\_ _ -> { model = {}, command = Cmd.none })\n        , subscriptions = (\\_ -> Sub.none)\n        }\n\n\ninit\n    : (Environment -> Init.Task { model : model, command : Cmd msg })\n    -> {}\n    -> { model : Model model, command : Cmd (Msg model msg) }\ninit initTask {} =\n    { model = Uninitialized\n    , command = \n        initializeEnvironment\n            |> Task.andThen (\\env -> unwrap <| initTask env)\n            |> Task.perform InitDone\n    }\n\n\nunwrap : Internal.Init.Task a -> Task Never a\nunwrap (Internal.Init.Task task) =\n    task\n\n\nupdate\n    : (msg -> model -> { model : model, command : Cmd msg })\n    -> Msg model msg\n    -> Model model\n    -> { model : Model model, command : Cmd (Msg model msg) }\nupdate appUpdate msg model =\n    case model of\n        Uninitialized ->\n            case msg of\n                InitDone initResult ->\n                    { model = Initialized initResult.model\n                    , command = Cmd.map MsgReceived initResult.command\n                    }\n\n                MsgReceived _ ->\n                    -- Ignore\n                    { model = model, command = Cmd.none }\n\n        Initialized appModel ->\n            case msg of\n                InitDone _ ->\n                    -- Ignore\n                    { model = model, command = Cmd.none }\n\n                MsgReceived appMsg ->\n                    let\n                        updateResult =\n                            appUpdate appMsg appModel\n                    in\n                    { model = Initialized updateResult.model\n                    , command = Cmd.map MsgReceived updateResult.command\n                    }\n\n\nsubscriptions\n    : (model -> Sub msg)\n    -> (Model model)\n    -> Sub (Msg model msg)\nsubscriptions appSubs model =\n    case model of\n        Uninitialized ->\n            Sub.none\n\n        Initialized appModel ->\n            Sub.map MsgReceived (appSubs appModel)\n\n\n{-| This lets the runtime know that you're done initializing other subsystems,\nand that your program is ready to start.\n-}\nstartProgram : { model : model, command : Cmd cmd } -> Init.Task { model : model, command : Cmd cmd }\nstartProgram initResult =\n    Internal.Init.Task (Task.succeed initResult)\n\n\n{-| When defining a program with [defineSimpleProgram](#defineSimpleProgram), use this function to define the\nfinal command to execute.\n-}\nendWithCmd : Cmd a -> Init.Task (Cmd a)\nendWithCmd cmd =\n    Internal.Init.Task (Task.succeed cmd)\n\n\n-- EXIT\n\n\n{-| Terminate the program immediatly. It will not wait for tasks like http calls\nor file system writes to complete.\n\nThis function is equivalent to:\n\n    exitWithCode 0\n-}\nexit : Task x {}\nexit =\n    exitWithCode 0\n\n\n{-| Terminate the program immediatly. It will not wait for tasks like http calls\nor file system writes, so only use this if you've reached a state where it makes\nno sense to continue.\n\nThe exit code can be read by other processes on your system. Any value other than\n0 is considered an error, but there are no other formal requirements for what\nmakes an exit code. If all you want is to signalize that your application exited\ndue to an error, -1 is a good option.\n-}\nexitWithCode : Int -> Task x {}\nexitWithCode code =\n    Gren.Kernel.Node.exitWithCode code\n\n\n{-| Set the error code that the program will return once it finishes.\n\nNote: This will not terminate your program, so things like http calls\nor writes to the filesystem will be allowed to complete. However,\nthe program will only exit once there are no ongoing tasks.\n-}\nsetExitCode : Int -> Task x {}\nsetExitCode code =\n    Gren.Kernel.Node.setExitCode code\n",
        "module Platform exposing\n    ( Program, worker\n    , Task, ProcessId\n    , Router, sendToApp, sendToSelf\n    )\n\n{-| This module contains definitions important to the language runtime.\nYou're unlikely to make direct use of these things yourself.\n\n\n@docs Program, worker\n\n\n## Tasks and Processes\n\n@docs Task, ProcessId\n\n\n## Effect Manager Helpers\n\nEffect managers can be viewed as programs-within-a-program. They have their own\nstate, and communicate with the application using messages.\n\nEffect managers are used internally for many things, but isn't considered to be\ntruly stable. It's likely that this feature will be redesigned in a future relase.\n\n\n@docs Router, sendToApp, sendToSelf\n\n-}\n\nimport Basics exposing (Never)\nimport Gren.Kernel.Platform\nimport Gren.Kernel.Scheduler\nimport Platform.Cmd exposing (Cmd)\nimport Platform.Sub exposing (Sub)\n\n\n\n-- PROGRAMS\n\n\n{-| A `Program` describes an Gren program! How does it react to input? Does it\nshow anything on screen? Etc.\n-}\ntype Program flags model msg\n    = Program\n\n\n{-| Create a [headless] program with no user interface.\n\nThis is great if you want to use Gren as the &ldquo;brain&rdquo; for something\nelse. For example, you could send messages out ports to modify the DOM, but do\nall the complex logic in Gren.\n\n[headless]: https://en.wikipedia.org/wiki/Headless_software\n\nInitializing a headless program from JavaScript looks like this:\n\n```javascript\nvar app = Gren.MyThing.init();\n```\n\nIf you _do_ want to control the user interface in Gren, the [`Browser`][browser]\nmodule has a few ways to create that kind of `Program` instead!\n\n[headless]: https://en.wikipedia.org/wiki/Headless_software\n[browser]: /package/gren-lang/browser/latest/module/Browser\n\n-}\nworker :\n    { init : flags -> { model : model, command : Cmd msg }\n    , update : msg -> model -> { model : model, command : Cmd msg }\n    , subscriptions : model -> Sub msg\n    }\n    -> Program flags model msg\nworker =\n    Gren.Kernel.Platform.worker\n\n\n\n-- TASKS and PROCESSES\n\n\n{-| Head over to the documentation for the [`Task`](Task) module for more\ninformation on this. It is only defined here because it is a platform\nprimitive.\n-}\ntype Task err ok\n    = Task\n\n\n{-| Head over to the documentation for the [`Process`](Process) module for\ninformation on this. It is only defined here because it is a platform\nprimitive.\n-}\ntype ProcessId\n    = ProcessId\n\n\n\n-- EFFECT MANAGER INTERNALS\n\n\n{-| An effect manager has access to a “router” that routes messages between\nthe main app and your individual effect manager.\n-}\ntype Router appMsg selfMsg\n    = Router\n\n\n{-| Send the router a message for the main loop of your app. This message will\nbe handled by the overall `update` function, just like events from `Html`.\n-}\nsendToApp : Router msg a -> msg -> Task x {}\nsendToApp =\n    Gren.Kernel.Platform.sendToApp\n\n\n{-| Send the router a message for your effect manager. This message will\nbe routed to the `onSelfMsg` function, where you can update the state of your\neffect manager as necessary.\n\nAs an example, the effect manager for web sockets\n\n-}\nsendToSelf : Router a msg -> msg -> Task x {}\nsendToSelf =\n    Gren.Kernel.Platform.sendToSelf\n",
        "module Platform.Sub exposing\n    ( Sub, none, batch\n    , map\n    )\n\n{-|\n\n> **Note:** Gren has **managed effects**, meaning that things like HTTP\n> requests or writing to disk are all treated as _data_ in Gren. When this\n> data is given to the Gren runtime system, it can do some “query optimization”\n> before actually performing the effect. Perhaps unexpectedly, this managed\n> effects idea is the heart of why Gren is so nice for testing, reuse,\n> reproducibility, etc.\n>\n> Gren has two kinds of managed effects: commands and subscriptions.\n\n\n## Subscriptions\n\n@docs Sub, none, batch\n\n\n## Fancy Stuff\n\n@docs map\n\n-}\n\nimport Array exposing (Array)\nimport Gren.Kernel.Platform\n\n\n\n-- SUBSCRIPTIONS\n\n\n{-| A subscription is a way of telling Gren, “Hey, let me know if anything\ninteresting happens over there!” So if you want to listen for messages on a web\nsocket, you would tell Gren to create a subscription. If you want to get clock\nticks, you would tell Gren to subscribe to that. The cool thing here is that\nthis means _Gren_ manages all the details of subscriptions instead of _you_.\nSo if a web socket goes down, _you_ do not need to manually reconnect with an\nexponential backoff strategy, _Gren_ does this all for you behind the scenes!\n\nEvery `Sub` specifies (1) which effects you need access to and (2) the type of\nmessages that will come back into your application.\n\n**Note:** Do not worry if this seems confusing at first! As with every Gren user\never, subscriptions will make more sense as you work through [the Gren Architecture\nTutorial](https://guide.gren-lang.org/architecture/) and see how they fit\ninto a real application!\n\n-}\ntype Sub msg\n    = Sub\n\n\n{-| Tell the runtime that there are no subscriptions.\n-}\nnone : Sub msg\nnone =\n    batch []\n\n\n{-| When you need to subscribe to multiple things, you can create a `batch` of\nsubscriptions.\n\n**Note:** `Sub.none` and `Sub.batch [ Sub.none, Sub.none ]` and\n`Sub.batch []` all do the same thing.\n\n-}\nbatch : Array (Sub msg) -> Sub msg\nbatch =\n    Gren.Kernel.Platform.batch\n\n\n\n-- FANCY STUFF\n\n\n{-| Transform the messages produced by a subscription.\nVery similar to [`Html.map`](/package/gren-lang/browser/latest/module/Html#map).\n\nThis is very rarely useful in well-structured Gren code, so definitely read the\nsection on [structure] in the guide before reaching for this!\n\n[structure]: https://guide.gren-lang.org/webapps/structure.html\n\n-}\nmap : (a -> msg) -> Sub a -> Sub msg\nmap =\n    Gren.Kernel.Platform.map\n",
        "module Platform.Cmd exposing\n    ( Cmd, none, batch\n    , map\n    )\n\n{-|\n\n> **Note:** Gren has **managed effects**, meaning that things like HTTP\n> requests or writing to disk are all treated as _data_ in Gren. When this\n> data is given to the Gren runtime system, it can do some “query optimization”\n> before actually performing the effect. Perhaps unexpectedly, this managed\n> effects idea is the heart of why Gren is so nice for testing, reuse,\n> reproducibility, etc.\n>\n> Gren has two kinds of managed effects: commands and subscriptions.\n\n\n## Commands\n\n@docs Cmd, none, batch\n\n\n## Fancy Stuff\n\n@docs map\n\n-}\n\nimport Array exposing (Array)\nimport Gren.Kernel.Platform\n\n\n\n-- COMMANDS\n\n\n{-| A command is a way of telling Gren, “Hey, I want you to do this thing!”\nSo if you want to send an HTTP request, you would need to command Gren to do it.\nOr if you wanted to ask for geolocation, you would need to command Gren to go\nget it.\n\nEvery `Cmd` specifies (1) which effects you need access to and (2) the type of\nmessages that will come back into your application.\n\n**Note:** Do not worry if this seems confusing at first! As with every Gren user\never, commands will make more sense as you work through [the Gren Architecture\nTutorial](https://guide.gren-lang.org/architecture/) and see how they\nfit into a real application!\n\n-}\ntype Cmd msg\n    = Cmd\n\n\n{-| Tell the runtime that there are no commands.\n-}\nnone : Cmd msg\nnone =\n    batch []\n\n\n{-| When you need the runtime system to perform a couple commands, you\ncan batch them together. Each is handed to the runtime at the same time,\nand since each can perform arbitrary operations in the world, there are\nno ordering guarantees about the results.\n\n**Note:** `Cmd.none` and `Cmd.batch [ Cmd.none, Cmd.none ]` and `Cmd.batch []`\nall do the same thing.\n\n-}\nbatch : Array (Cmd msg) -> Cmd msg\nbatch =\n    Gren.Kernel.Platform.batch\n\n\n\n-- FANCY STUFF\n\n\n{-| Transform the messages produced by a command.\nVery similar to [`Html.map`](/package/gren-lang/browser/latest/module/Html#map).\n\nThis is very rarely useful in well-structured Gren code, so definitely read the\nsection on [structure] in the guide before reaching for this!\n\n[structure]: https://guide.gren-lang.org/webapps/structure.html\n\n-}\nmap : (a -> msg) -> Cmd a -> Cmd msg\nmap =\n    Gren.Kernel.Platform.map\n",
        "module Init exposing\n    ( Task\n    , await\n    , awaitTask\n    )\n\n{-| This module defines the app initialization task. This is a special kind of task\nthat can only be passed as the result of an application initialization. You'll typically\nuse this module in order to initialize sub-systems like `FileSystem` or `ChildProcess`.\n\n@docs Task, await, awaitTask\n-}\n\n\nimport Task\nimport Internal.Init\n\n\n{-| This is like a `Task`, but can only be run as part of initializing your\nprogram. This is usually used for values which should be provided to your program,\nand only your program, as opposed to third-party packages.\n-}\ntype alias Task a =\n    Internal.Init.Task a\n\n\n{-| This let's you wait for the completion of an `Task` before either starting\nyour application, or begin initialization of another `Task`.\n\n    Init.await Terminal.initialize <| \\termConfig ->\n    Init.await FileSystem.initialize <| \\fileSystemConfig ->\n        -- Start your own program with the values from Terminal and FileSystem\n        Node.startProgram\n            { model = 1\n            , commands = Cmd.none\n            }\n-}\nawait : Task a -> (a -> Task b) -> Task b\nawait (Internal.Init.Task task) fn =\n    Internal.Init.Task (Task.andThen (unwrap << fn) task)\n\n\n{-| This let's you wait for the completion of a `Task` before either starting\nyour application, or begin initialization of another `Task`.\n\n    Init.await Terminal.initialize <| \\termConfig ->\n    Init.awaitTask Task.now <| \\time ->\n        -- Start your own program with the values from Terminal and FileSystem\n        Node.startProgram\n            { model = time\n            , commands = Cmd.none\n            }\n-}\nawaitTask : Task.Task Never a -> (a -> Task b) -> Task b\nawaitTask task fn =\n    Internal.Init.Task (Task.andThen (unwrap << fn) task)\n\n\nunwrap : Task a -> Task.Task Never a\nunwrap (Internal.Init.Task task) =\n    task\n\n\n",
        "effect module Time where { subscription = MySub } exposing\n  ( Posix\n  , now\n  , every\n  , posixToMillis\n  , millisToPosix\n  , Zone\n  , utc\n  , here\n  , toYear\n  , toMonth\n  , toDay\n  , toWeekday\n  , toHour\n  , toMinute\n  , toSecond\n  , toMillis\n  , Month(..)\n  , Weekday(..)\n  , customZone\n  , getZoneName\n  , ZoneName(..)\n  )\n\n\n{-| Functions for working with time and time zones.\n\n@docs Posix, now, every, posixToMillis, millisToPosix\n\n## Time Zones\n@docs Zone, utc, here\n\n## Human Times\n@docs toYear, toMonth, toDay, toWeekday, toHour, toMinute, toSecond, toMillis\n\n## Weeks and Months\n@docs Weekday, Month\n\n## For Package Authors\n@docs customZone, getZoneName, ZoneName\n\n-}\n\n\nimport Array exposing (Array)\nimport Basics exposing (..)\nimport Dict\nimport Math\nimport Maybe exposing (Maybe(..))\nimport Platform\nimport Platform.Sub exposing (Sub)\nimport Process\nimport String exposing (String)\nimport Task exposing (Task)\nimport Gren.Kernel.Time\n\n\n\n-- POSIX\n\n\n{-| A computer representation of time. It is the same all over Earth, so if we\nhave a phone call or meeting at a certain POSIX time, there is no ambiguity.\n\nIt is very hard for humans to _read_ a POSIX time though, so we use functions\nlike [`toHour`](#toHour) and [`toMinute`](#toMinute) to `view` them.\n-}\ntype Posix = Posix Int\n\n\n{-| Get the POSIX time at the moment when this task is run.\n-}\nnow : Task x Posix\nnow =\n  Gren.Kernel.Time.now millisToPosix\n\n\n{-| Turn a `Posix` time into the number of milliseconds since 1970 January 1\nat 00:00:00 UTC. It was a Thursday.\n-}\nposixToMillis : Posix -> Int\nposixToMillis (Posix millis) =\n  millis\n\n\n{-| Turn milliseconds into a `Posix` time.\n-}\nmillisToPosix : Int -> Posix\nmillisToPosix =\n  Posix\n\n\n\n-- TIME ZONES\n\n\n{-| Information about a particular time zone.\n\nThe [IANA Time Zone Database][iana] tracks things like UTC offsets and\ndaylight-saving rules so that you can turn a `Posix` time into local times\nwithin a time zone.\n\nSee [`utc`](#utc) and [`here`](#here) to learn how to obtain `Zone` values.\n\n[iana]: https://www.iana.org/time-zones\n-}\ntype Zone =\n  Zone Int (Array Era)\n\n\n-- TODO: add this note back to `Zone` docs when it is true\n--\n-- Did you know that in California the times change from 3pm PST to 3pm PDT to\n-- capture whether it is daylight-saving time? The database tracks those\n-- abbreviation changes too. (Tons of time zones do that actually.)\n--\n\n\n{-| Currently the public API only needs:\n\n- `start` is the beginning of this `Era` in \"minutes since the Unix Epoch\"\n- `offset` is the UTC offset of this `Era` in minutes\n\nBut eventually, it will make sense to have `abbr : String` for `PST` vs `PDT`\n-}\ntype alias Era =\n  { start : Int\n  , offset : Int\n  }\n\n\n{-| The time zone for Coordinated Universal Time ([UTC][])\n\nThe `utc` zone has no time adjustments. It never observes daylight-saving\ntime and it never shifts around based on political restructuring.\n\n[UTC]: https://en.wikipedia.org/wiki/Coordinated_Universal_Time\n-}\nutc : Zone\nutc =\n  Zone 0 []\n\n\n{-| Produce a `Zone` based on the current UTC offset. You can use this to figure\nout what day it is where you are:\n\n    import Task exposing (Task)\n    import Time\n\n    whatDayIsIt : Task x Int\n    whatDayIsIt =\n      Task.map2 Time.toDay Time.here Time.now\n\n**Accuracy Note:** This function can only give time zones like `Etc/GMT+9` or\n`Etc/GMT-6`. It cannot give you `Europe/Stockholm`, `Asia/Tokyo`, or any other\nnormal time zone from the [full list][tz] due to limitations in JavaScript.\nFor example, if you run `here` in New York City, the resulting `Zone` will\nnever be `America/New_York`. Instead you get `Etc/GMT-5` or `Etc/GMT-4`\ndepending on Daylight Saving Time. So even though browsers must have internal\naccess to `America/New_York` to figure out that offset, there is no public API\nto get the full information. This means the `Zone` you get from this function\nwill act weird if (1) an application stays open across a Daylight Saving Time\nboundary or (2) you try to use it on historical data.\n\n**Future Note:** We can improve `here` when there is good browser support for\nJavaScript functions that (1) expose the IANA time zone database and (2) let\nyou ask the time zone of the computer. The committee that reviews additions to\nJavaScript is called TC39, and I encourage you to push for these capabilities! I\ncannot do it myself unfortunately.\n\n**Alternatives:** See the `customZone` docs to learn how to implement stopgaps.\n\n[tz]: https://en.wikipedia.org/wiki/List_of_tz_database_time_zones\n-}\nhere : Task x Zone\nhere =\n  Gren.Kernel.Time.here {}\n\n\n\n-- DATES\n\n\n{-| What year is it?!\n\n    import Time exposing (toYear, utc, millisToPosix)\n\n    toYear utc (millisToPosix 0) == 1970\n    toYear nyc (millisToPosix 0) == 1969\n\n    -- pretend `nyc` is the `Zone` for America/New_York.\n-}\ntoYear : Zone -> Posix -> Int\ntoYear zone time =\n  (toCivil (toAdjustedMinutes zone time)).year\n\n\n{-| What month is it?!\n\n    import Time exposing (toMonth, utc, millisToPosix)\n\n    toMonth utc (millisToPosix 0) == Jan\n    toMonth nyc (millisToPosix 0) == Dec\n\n    -- pretend `nyc` is the `Zone` for America/New_York.\n-}\ntoMonth : Zone -> Posix -> Month\ntoMonth zone time =\n  case (toCivil (toAdjustedMinutes zone time)).month of\n    1  -> Jan\n    2  -> Feb\n    3  -> Mar\n    4  -> Apr\n    5  -> May\n    6  -> Jun\n    7  -> Jul\n    8  -> Aug\n    9  -> Sep\n    10 -> Oct\n    11 -> Nov\n    _  -> Dec\n\n\n{-| What day is it?! (Days go from 1 to 31)\n\n    import Time exposing (toDay, utc, millisToPosix)\n\n    toDay utc (millisToPosix 0) == 1\n    toDay nyc (millisToPosix 0) == 31\n\n    -- pretend `nyc` is the `Zone` for America/New_York.\n\n-}\ntoDay : Zone -> Posix -> Int\ntoDay zone time =\n  (toCivil (toAdjustedMinutes zone time)).day\n\n\n{-| What day of the week is it?\n\n    import Time exposing (toWeekday, utc, millisToPosix)\n\n    toWeekday utc (millisToPosix 0) == Thu\n    toWeekday nyc (millisToPosix 0) == Wed\n\n    -- pretend `nyc` is the `Zone` for America/New_York.\n-}\ntoWeekday : Zone -> Posix -> Weekday\ntoWeekday zone time =\n  case Math.modBy 7 (flooredDiv (toAdjustedMinutes zone time) (60 * 24)) of\n    0 -> Thu\n    1 -> Fri\n    2 -> Sat\n    3 -> Sun\n    4 -> Mon\n    5 -> Tue\n    _ -> Wed\n\n\n{-| What hour is it? (From 0 to 23)\n\n    import Time exposing (toHour, utc, millisToPosix)\n\n    toHour utc (millisToPosix 0) == 0  -- 12am\n    toHour nyc (millisToPosix 0) == 19 -- 7pm\n\n    -- pretend `nyc` is the `Zone` for America/New_York.\n-}\ntoHour : Zone -> Posix -> Int\ntoHour zone time =\n  Math.modBy 24 (flooredDiv (toAdjustedMinutes zone time) 60)\n\n\n{-| What minute is it? (From 0 to 59)\n\n    import Time exposing (toMinute, utc, millisToPosix)\n\n    toMinute utc (millisToPosix 0) == 0\n\nThis can be different in different time zones. Some time zones are offset\nby 30 or 45 minutes!\n-}\ntoMinute : Zone -> Posix -> Int\ntoMinute zone time =\n  Math.modBy 60 (toAdjustedMinutes zone time)\n\n\n{-| What second is it?\n\n    import Time exposing (toSecond, utc, millisToPosix)\n\n    toSecond utc (millisToPosix    0) == 0\n    toSecond utc (millisToPosix 1234) == 1\n    toSecond utc (millisToPosix 5678) == 5\n-}\ntoSecond : Zone -> Posix -> Int\ntoSecond _ time =\n  Math.modBy 60 (flooredDiv (posixToMillis time) 1000)\n\n\n{-|\n    import Time exposing (toMillis, utc, millisToPosix)\n\n    toMillis utc (millisToPosix    0) == 0\n    toMillis utc (millisToPosix 1234) == 234\n    toMillis utc (millisToPosix 5678) == 678\n-}\ntoMillis : Zone -> Posix -> Int\ntoMillis _ time =\n  Math.modBy 1000 (posixToMillis time)\n\n\n\n-- DATE HELPERS\n\n\ntoAdjustedMinutes : Zone -> Posix -> Int\ntoAdjustedMinutes (Zone defaultOffset eras) time =\n  toAdjustedMinutesHelp defaultOffset (flooredDiv (posixToMillis time) 60000) eras\n\n\ntoAdjustedMinutesHelp : Int -> Int -> Array Era -> Int\ntoAdjustedMinutesHelp defaultOffset posixMinutes eras =\n  case Array.get 0 eras of\n    Nothing ->\n      posixMinutes + defaultOffset\n\n    Just era ->\n      if era.start < posixMinutes then\n        posixMinutes + era.offset\n      else\n        toAdjustedMinutesHelp defaultOffset posixMinutes (Array.slice 1 (Array.length eras) eras)\n\n\ntoCivil : Int -> { year : Int, month : Int, day : Int }\ntoCivil minutes =\n  let\n    rawDay    = flooredDiv minutes (60 * 24) + 719468\n    era       = (if rawDay >= 0 then rawDay else rawDay - 146096) // 146097\n    dayOfEra  = rawDay - era * 146097 -- [0, 146096]\n    yearOfEra = (dayOfEra - dayOfEra // 1460 + dayOfEra // 36524 - dayOfEra // 146096) // 365 -- [0, 399]\n    year      = yearOfEra + era * 400\n    dayOfYear = dayOfEra - (365 * yearOfEra + yearOfEra // 4 - yearOfEra // 100) -- [0, 365]\n    mp        = (5 * dayOfYear + 2) // 153 -- [0, 11]\n    month     = mp + (if mp < 10 then 3 else -9) -- [1, 12]\n  in\n  { year = year + (if month <= 2 then 1 else 0)\n  , month = month\n  , day = dayOfYear - (153 * mp + 2) // 5 + 1 -- [1, 31]\n  }\n\n\nflooredDiv : Int -> Float -> Int\nflooredDiv numerator denominator =\n  Math.floor (toFloat numerator / denominator)\n\n\n\n-- WEEKDAYS AND MONTHS\n\n\n{-| Represents a `Weekday` so that you can convert it to a `String` or `Int`\nhowever you please. For example, if you need the Japanese representation, you\ncan say:\n\n    toJapaneseWeekday : Weekday -> String\n    toJapaneseWeekday weekday =\n      case weekday of\n        Mon -> \"月\"\n        Tue -> \"火\"\n        Wed -> \"水\"\n        Thu -> \"木\"\n        Fri -> \"金\"\n        Sat -> \"土\"\n        Sun -> \"日\"\n-}\ntype Weekday = Mon | Tue | Wed | Thu | Fri | Sat | Sun\n\n\n{-| Represents a `Month` so that you can convert it to a `String` or `Int`\nhowever you please. For example, if you need the Danish representation, you\ncan say:\n\n    toDanishMonth : Month -> String\n    toDanishMonth month =\n      case month of\n        Jan -> \"januar\"\n        Feb -> \"februar\"\n        Mar -> \"marts\"\n        Apr -> \"april\"\n        May -> \"maj\"\n        Jun -> \"juni\"\n        Jul -> \"juli\"\n        Aug -> \"august\"\n        Sep -> \"september\"\n        Oct -> \"oktober\"\n        Nov -> \"november\"\n        Dec -> \"december\"\n-}\ntype Month = Jan | Feb | Mar | Apr | May | Jun | Jul | Aug | Sep | Oct | Nov | Dec\n\n\n\n-- SUBSCRIPTIONS\n\n\n{-| Get the current time periodically. How often though? Well, you provide an\ninterval in milliseconds (like `1000` for a second or `60 * 1000` for a minute\nor `60 * 60 * 1000` for an hour) and that is how often you get a new time!\n\nCheck out [this example](https://gren-lang.org/examples/time) to see how to use\nit in an application.\n\n**This function is not for animation.** Use the [`onAnimationFrame`][af]\nfunction for that sort of thing! It syncs up with repaints and will end up\nbeing much smoother for any moving visuals.\n\n[af]: /packages/gren-lang/browser/latest/Browser-Events#onAnimationFrame\n-}\nevery : Float -> (Posix -> msg) -> Sub msg\nevery interval tagger =\n  subscription (Every interval tagger)\n\n\ntype MySub msg =\n  Every Float (Posix -> msg)\n\n\nsubMap : (a -> b) -> MySub a -> MySub b\nsubMap f (Every interval tagger) =\n  Every interval (f << tagger)\n\n\n\n-- EFFECT MANAGER\n\n\ntype alias State msg =\n  { taggers : Taggers msg\n  , processes : Processes\n  }\n\n\ntype alias Processes =\n  Dict.Dict Float Platform.ProcessId\n\n\ntype alias Taggers msg =\n  Dict.Dict Float (Array (Posix -> msg))\n\n\ninit : Task Never (State msg)\ninit =\n  Task.succeed { taggers = Dict.empty, processes = Dict.empty }\n\n\nonEffects : Platform.Router msg Float -> Array (MySub msg) -> State msg -> Task Never (State msg)\nonEffects router subs {processes} =\n  let\n    newTaggers =\n      Array.foldl addMySub Dict.empty subs\n\n    spawnArray =\n        Dict.diff newTaggers processes\n            |> Dict.keys\n\n    existingDict =\n        Dict.filter (\\key _ -> Dict.member key newTaggers) processes\n\n    killTask =\n        Dict.diff processes newTaggers\n            |> Dict.values\n            |> Array.foldl (\\id -> Task.andThen (\\_ -> Process.kill id)) (Task.succeed {})\n  in\n    killTask\n      |> Task.andThen (\\_ -> spawnHelp router spawnArray existingDict)\n      |> Task.andThen (\\newProcesses -> Task.succeed { taggers = newTaggers, processes = newProcesses })\n\n\naddMySub : MySub msg -> Taggers msg -> Taggers msg\naddMySub (Every interval tagger) state =\n  case Dict.get interval state of\n    Nothing ->\n      Dict.set interval [tagger] state\n\n    Just taggers ->\n      Dict.set interval ([tagger] ++ taggers) state\n\n\nspawnHelp : Platform.Router msg Float -> Array Float -> Processes -> Task.Task x Processes\nspawnHelp router intervals processes =\n  case Array.get 0 intervals of\n    Nothing ->\n      Task.succeed processes\n\n    Just interval ->\n      let\n        spawnTimer =\n          Process.spawn (setInterval interval (Platform.sendToSelf router interval))\n\n        rest =\n          Array.slice 1 (Array.length intervals) intervals\n\n        spawnRest id =\n          spawnHelp router rest (Dict.set interval id processes)\n      in\n        spawnTimer\n          |> Task.andThen spawnRest\n\n\nonSelfMsg : Platform.Router msg Float -> Float -> State msg -> Task Never (State msg)\nonSelfMsg router interval state =\n  case Dict.get interval state.taggers of\n    Nothing ->\n      Task.succeed state\n\n    Just taggers ->\n      let\n        tellTaggers time =\n          Task.sequence (Array.map (\\tagger -> Platform.sendToApp router (tagger time)) taggers)\n      in\n        now\n          |> Task.andThen tellTaggers\n          |> Task.andThen (\\_ -> Task.succeed state)\n\n\nsetInterval : Float -> Task Never {} -> Task x Never\nsetInterval =\n  Gren.Kernel.Time.setInterval\n\n\n\n-- FOR PACKAGE AUTHORS\n\n\n\n{-| **Intended for package authors.**\n\nThe documentation of [`here`](#here) explains that it has certain accuracy\nlimitations that block on adding new APIs to JavaScript. The `customZone`\nfunction is a stopgap that takes:\n\n1. A default offset in minutes. So `Etc/GMT-5` is `customZone (-5 * 60) []`\nand `Etc/GMT+9` is `customZone (9 * 60) []`.\n2. A list of exceptions containing their `start` time in \"minutes since the Unix\nepoch\" and their `offset` in \"minutes from UTC\"\n\nHuman times will be based on the nearest `start`, falling back on the default\noffset if the time is older than all of the exceptions.\n\nWhen paired with `getZoneName`, this allows you to load the real IANA time zone\ndatabase however you want: HTTP, cache, hardcode, etc.\n\n**Note:** If you use this, please share your work in an Gren community forum!\nI am sure others would like to hear about it, and more experience reports will\nhelp me and the any potential TC39 proposal.\n-}\ncustomZone : Int -> Array { start : Int, offset : Int } -> Zone\ncustomZone =\n  Zone\n\n\n{-| **Intended for package authors.**\n\nUse `Intl.DateTimeFormat().resolvedOptions().timeZone` to try to get names\nlike `Europe/Moscow` or `America/Havana`. From there you can look it up in any\nIANA data you loaded yourself.\n-}\ngetZoneName : Task x ZoneName\ngetZoneName =\n  Gren.Kernel.Time.getZoneName {}\n\n\n{-| **Intended for package authors.**\n\nThe `getZoneName` function relies on a JavaScript API that is not supported\nin all browsers yet, so it can return the following:\n\n    -- in more recent browsers\n    Name \"Europe/Moscow\"\n    Name \"America/Havana\"\n\n    -- in older browsers\n    Offset 180\n    Offset -300\n\nSo if the real info is not available, it will tell you the current UTC offset\nin minutes, just like what `here` uses to make zones like `customZone -60 []`.\n-}\ntype ZoneName\n  = Name String\n  | Offset Int\n",
        "effect module FileSystem where { subscription = FileSystemSub } exposing\n    ( Permission\n    , initialize\n    -- Errors\n    , Error\n    , errorCode\n    , errorToString\n    , errorIsPermissionDenied\n    , errorIsFileExists\n    , errorIsDirectoryFound\n    , errorIsTooManyOpenFiles\n    , errorIsNoSuchFileOrDirectory\n    , errorIsNotADirectory\n    , errorIsDirectoryNotEmpty\n    , errorIsNotPermitted\n    , errorIsLinkLoop\n    , errorIsPathTooLong\n    , errorIsInvalidInput\n    , errorIsIO\n    -- Metadata\n    , Metadata\n    , EntityType(..)\n    , metadata\n    , AccessPermission(..)\n    , checkAccess\n    , changeAccess\n    , accessPermissionsToInt\n    , changeOwner\n    , changeTimes\n    , move\n    , realPath\n    -- Files\n    , appendToFile\n    , copyFile\n    , readFile\n    , writeFile\n    , truncateFile\n    , remove\n    -- Directories\n    , listDirectory\n    , makeDirectory\n    , makeTempDirectory\n    -- Links\n    , hardLink\n    , softLink\n    , readLink\n    , unlink\n    -- Watch\n    , WatchEvent(..)\n    , watch\n    , watchRecursive\n    --\n    , homeDirectory\n    , currentWorkingDirectory\n    , tmpDirectory\n    , devNull\n    )\n\n\n{-| This module provides access to the file system. It allows you to read and write files, create directories and links etc.\n\n@docs Permission, initialize\n\n## Errors\n\n@docs Error, errorCode, errorToString\n@docs errorIsPermissionDenied, errorIsFileExists, errorIsDirectoryFound, errorIsTooManyOpenFiles, errorIsNoSuchFileOrDirectory, errorIsNotADirectory, errorIsDirectoryNotEmpty, errorIsNotPermitted, errorIsLinkLoop, errorIsPathTooLong, errorIsInvalidInput, errorIsIO\n\n## Metadata\n\n@docs Metadata, EntityType, metadata, AccessPermission, checkAccess, changeAccess, accessPermissionsToInt, changeOwner, changeTimes, move, realPath\n\n## Files\n\n@docs copyFile, appendToFile, readFile, writeFile, truncateFile, remove\n\n## Directories\n\n@docs listDirectory, makeDirectory, makeTempDirectory \n\n## Links\n\n@docs hardLink, softLink, readLink, unlink\n\n## Watch for changes\n\n@docs WatchEvent, watch, watchRecursive\n\n## Special paths\n\n@docs homeDirectory, currentWorkingDirectory, tmpDirectory, devNull\n-}\n\n\nimport Gren.Kernel.FileSystem\nimport Gren.Kernel.FilePath\nimport Bytes exposing (Bytes)\nimport Dict exposing (Dict)\nimport Task exposing (Task)\nimport FileSystem.Path exposing (Path)\nimport Init\nimport Internal.Init\nimport Time\nimport Process\n\n\n{-| This value represents the permission to perform file system operations.\n\nOnly code you trust should have access to this value.\n-}\ntype Permission\n    = Permission\n\n\n{-| Initialize the `FileSystem` subsystem, which gains you the permission to perform\nfile system operations.\n-}\ninitialize : Init.Task Permission\ninitialize =\n    Task.succeed Permission\n        |> Internal.Init.Task\n\n\n-- ERRORS\n\n\n{-| Represents an error that occured when working with the file system.\n\nThere are many different kinds of error depending on which operation you're performing and which\noperating system you're performing it on. To figure out which error it is, you'll need\nto use one of the helper functions below, or check the specific error code.\n-}\ntype Error\n    = Error String String\n\n\n{-| A string that identifies a specific kind of error. There can be several error codes for the\nsame kind of error, depending on the operating system that is in use.\n-}\nerrorCode : Error -> String\nerrorCode (Error code _) =\n    code\n\n\n{-| Returns a human readable description of the error.\n-}\nerrorToString : Error -> String\nerrorToString (Error _ message) =\n    message\n\n\n{-| If `True`, the error occured because you don't have the correct access permission to perform\nthe operation.\n-}\nerrorIsPermissionDenied : Error -> Bool\nerrorIsPermissionDenied (Error code _) =\n    code == \"EACCES\"\n\n\n{-| If `True`, a file exists when it was expected not to.\n-}\nerrorIsFileExists : Error -> Bool\nerrorIsFileExists (Error code _) =\n    code == \"EEXIST\"\n\n\n{-| If `True`, a file operation was attempted on a directory.\n-}\nerrorIsDirectoryFound : Error -> Bool\nerrorIsDirectoryFound (Error code _) =\n    code == \"EISDIR\"\n\n\n{-| If `True`, the application has too many open files.\n-}\nerrorIsTooManyOpenFiles : Error -> Bool\nerrorIsTooManyOpenFiles (Error code _) =\n    code == \"EMFILE\"\n\n\n{-| If `True`, the code was passed a [Path](FileSystem.Path#Path) which points to a file or directory\nthat doesn't exist.\n-}\nerrorIsNoSuchFileOrDirectory : Error -> Bool\nerrorIsNoSuchFileOrDirectory (Error code _) =\n    code == \"ENOENT\"\n\n\n{-| If `True`, a directory was expected but it found a file or some other entity.\n-}\nerrorIsNotADirectory : Error -> Bool\nerrorIsNotADirectory (Error code _) =\n    code == \"ENOTDIR\"\n\n\n{-| If `True`, the operation expected an empty directory, but the directory is not empty.\n-}\nerrorIsDirectoryNotEmpty : Error -> Bool\nerrorIsDirectoryNotEmpty (Error code _) =\n    code == \"ENOTEMPTY\"\n\n\n{-| If `True`, the operation was rejected because of missing privileges.\n-}\nerrorIsNotPermitted : Error -> Bool\nerrorIsNotPermitted (Error code _) =\n    code == \"EPERM\"\n\n\n{-| If `True`, we seem to be stuck in a loop following link after link after...\n-}\nerrorIsLinkLoop : Error -> Bool\nerrorIsLinkLoop (Error code _) =\n    code == \"ELOOP\"\n\n\n{-| If `True`, the [Path](FileSystem.Path#Path) is too long.\n-}\nerrorIsPathTooLong : Error -> Bool\nerrorIsPathTooLong (Error code _) =\n    code == \"ENAMETOOLONG\"\n\n\n{-| If `True`, the arguments passed to the function is invalid somehow.\n-}\nerrorIsInvalidInput : Error -> Bool\nerrorIsInvalidInput (Error code _) =\n    code == \"EINVAL\"\n\n\n{-| If `True`, the operation failed due to an IO error. This could be that the disk is\nbusy, or even corrupt.\n-}\nerrorIsIO : Error -> Bool\nerrorIsIO (Error code _) =\n    code == \"EIO\"\n\n\n-- METADATA\n\n\n{-| Represents extra information about an entity in the file system.\n-}\ntype alias Metadata =\n    { entityType : EntityType\n    , deviceID : Int\n    , userID : Int\n    , groupID : Int\n    , byteSize : Int\n    , blockSize : Int\n    , blocks : Int\n    , lastAccessed : Time.Posix\n    , lastModified : Time.Posix\n    , lastChanged : Time.Posix\n    , created : Time.Posix\n    }\n\n\n{-| The type of an entity in the file system.\n-}\ntype EntityType\n    = File\n    | Directory\n    | Socket\n    | Symlink\n    | Device\n    | Pipe\n\n\n{-| Return metadata for the entity represented by [Path](FileSystem.Path#Path).\n\nIf `resolveLink` is `False`, you will receive metadata for the link itself, not the entity\npointed at by the link.\n-}\nmetadata : Permission -> { resolveLink : Bool } -> Path -> Task Error Metadata\nmetadata _ { resolveLink } path =\n    if resolveLink then\n        Gren.Kernel.FileSystem.stat path\n\n    else\n        Gren.Kernel.FileSystem.lstat path\n\n\n{-| Represents the permission to access an entity for a specific operation.\n\nFor example: if you, or your group, doesn't have the `Read` permission for a file,\nyou're not allowed to read from it.\n-}\ntype AccessPermission\n    = Read\n    | Write\n    | Execute\n\n\n{-| Check if the user running this application has the given access permissions for the\nentity represented by [Path](FileSystem.Path#Path).\n\nPassing an empty `Array` will check that the entity exists.\n-}\ncheckAccess : Permission -> Array AccessPermission -> Path -> Task Error Path\ncheckAccess _ permissions path =\n    Gren.Kernel.FileSystem.access permissions path\n\n\n{-| Change the access permissions for the entity's owner, group and everyone else.\n-}\nchangeAccess \n    : Permission\n    -> { owner : Array AccessPermission \n       , group : Array AccessPermission\n       , others : Array AccessPermission\n       }\n    -> Path\n    -> Task Error Path\nchangeAccess _ permissions path =\n    let\n        mode =\n            (String.fromInt <| accessPermissionsToInt permissions.owner)\n            ++ (String.fromInt <| accessPermissionsToInt permissions.group)\n            ++ (String.fromInt <| accessPermissionsToInt permissions.others)\n    in\n    Gren.Kernel.FileSystem.chmod mode path\n\n\n{-| The integer representation of a set of access permissions in a posix system.\n\n    accessPermissionsToInt [ Read, Write ] == 6\n-}\naccessPermissionsToInt : Array AccessPermission -> Int\naccessPermissionsToInt values =\n    let\n        numberFor num a =\n            if Array.member a values then\n                num\n\n            else\n                0\n    in\n    numberFor 4 Read + numberFor 2 Write + numberFor 1 Execute\n\n\n{-| Change the user and group that owns a file.\n\nYou'll need the ID of the owner and group to perform this operation.\n\nIf `resolveLink` is `False`, you're changing the owner of the link itself,\nnot the entity it points to.\n-}\nchangeOwner : Permission -> { userID : Int, groupID : Int, resolveLink : Bool } -> Path -> Task Error Path\nchangeOwner _ options path =\n    if options.resolveLink then\n        Gren.Kernel.FileSystem.chown options path\n\n    else\n        Gren.Kernel.FileSystem.lchown options path\n\n\n{-| Change the registered time (down to the second) an entity was accessed and modified.\n\nIf `resolveLink` is `False`, you're changing the last access and modification time of the link itself,\nnot the entity it points to.\n-}\nchangeTimes : Permission -> { lastAccessed : Time.Posix, lastModified : Time.Posix, resolveLink : Bool } -> Path -> Task Error Path\nchangeTimes _ { lastAccessed, lastModified, resolveLink } path =\n    let\n        lastAccessedSeconds = \n            Time.posixToMillis lastAccessed // 1000\n        \n        lastModifiedSeconds = \n            Time.posixToMillis lastModified // 1000\n    in\n    if resolveLink then\n        Gren.Kernel.FileSystem.utimes lastAccessedSeconds lastModifiedSeconds path\n\n    else\n        Gren.Kernel.FileSystem.lutimes lastAccessedSeconds lastModifiedSeconds path\n\n\n{-| Move the entity represented by the last [Path](FileSystem.Path#Path), to the location\nrepresented by the first [Path](FileSystem.Path#Path). This can also be used to rename an\nentity.\n-}\nmove : Permission -> Path -> Path -> Task Error Path\nmove _ new old =\n    Gren.Kernel.FileSystem.rename old new\n\n\n{-| If you have a [Path](FileSystem.Path#Path) that is relative to the current directory,\nor points at a link, you can use this find the true [Path](FileSystem.Path#Path) of the\nentity.\n-}\nrealPath : Permission -> Path -> Task Error Path\nrealPath _ path =\n    Gren.Kernel.FileSystem.realpath path\n\n\n-- FILES\n\n\n{-| Add `Bytes` to the end of a file.\n-}\nappendToFile : Permission -> Bytes -> Path -> Task Error Path\nappendToFile _ bytes path =\n    Gren.Kernel.FileSystem.appendFile bytes path\n\n\n{-| Copy the file represented by the last [Path](FileSystem.Path#Path), to the location\nrepresented by the first [Path](FileSystem.Path#Path).\n-}\ncopyFile : Permission -> Path -> Path -> Task Error Path\ncopyFile _ dest src =\n    Gren.Kernel.FileSystem.copyFile src dest\n\n\n{-| Read the entire contents of a file.\n-}\nreadFile : Permission -> Path -> Task Error Bytes\nreadFile _ path =\n    Gren.Kernel.FileSystem.readFile path\n\n\n{-| Write the given `Bytes` into a file. The file will be created if it doesn't exist,\nand overwritten if it does.\n-}\nwriteFile : Permission -> Bytes -> Path -> Task Error Path\nwriteFile _ bytes path =\n    Gren.Kernel.FileSystem.writeFile bytes path\n\n\n{-| Make sure the given file is of a specific length. If the file is smaller than\nthe given length, zeroes is added to the file until it is the correct length. If the file\nis larger than the given length, the excess bytes are removed.\n-}\ntruncateFile : Permission -> Int -> Path -> Task Error Path\ntruncateFile _ length path =\n    Gren.Kernel.FileSystem.truncate length path\n\n\n{-| Remove the file or directory at the given path.\n\n* `recursive` will delete everything inside a directory.\n* `ignoreErrors` will... ignore any errors related to a remove operation.\n-}\nremove : Permission -> { recursive : Bool, ignoreErrors : Bool } -> Path -> Task Error Path\nremove _ options path =\n    Gren.Kernel.FileSystem.remove options path\n\n\n-- DIRECTORIES\n\n\n{-| List the contents of a directory. The returned [Paths](FileSystem.Path#Path) are relative to\nthe directory being listed.\n-}\nlistDirectory : Permission -> Path -> Task Error (Array { path : Path, entityType : EntityType })\nlistDirectory _ path =\n    Gren.Kernel.FileSystem.listDirectory path\n\n\n{-| Create a new directory at the given [Path](FileSystem.Path#Path).\n\nIf `recursive` is `True`, then a directory will be created for every section of the\ngiven [Path](FileSystem.Path#Path).\n-}\nmakeDirectory : Permission -> { recursive : Bool } -> Path -> Task Error Path\nmakeDirectory _ options path =\n    Gren.Kernel.FileSystem.makeDirectory options path\n\n\n{-| Create a directory, prefixed by a given name, that ends up in a section of the\nfile system reserved for temporary files. You're given the [Path](FileSystem.Path#Path)\nto this new directory.\n-}\nmakeTempDirectory : Permission -> String -> Task Error Path\nmakeTempDirectory _ prefix =\n    Gren.Kernel.FileSystem.mkdtemp prefix\n\n\n-- LINKS\n\n\n{-| Creates a hard link from the last [Path](FileSystem.Path#Path) to\nthe first.\n\nA hard link is an alias for a specific location. The link has the same\nownership and access permissions, and it's impossible to tell which is\nthe \"real\" entity and which is the link.\n-}\nhardLink : Permission -> Path -> Path -> Task Error Path\nhardLink _ dest src =\n    Gren.Kernel.FileSystem.link src dest\n\n\n{-| Creates a soft link from the last [Path](FileSystem.Path#Path) to\nthe first.\n\nA soft link, also known as a symoblic link or symlink, is a special file\nthat contains the path to some other location. Resolving a soft link will\nredirect to this other location.\n-}\nsoftLink : Permission -> Path -> Path -> Task Error Path\nsoftLink _ dest src =\n    Gren.Kernel.FileSystem.symlink src dest\n\n\n{-| Returns the [Path](FileSystem.Path#Path) pointed to by a soft link.\n-}\nreadLink : Permission -> Path -> Task Error Path\nreadLink _ path =\n    Gren.Kernel.FileSystem.readLink path\n\n\n{-| Removes a link, hard or soft, from the file system. If the\n[Path](FileSystem.Path#Path) refers to a file, the file is removed.\n-}\nunlink : Permission -> Path -> Task Error Path\nunlink _ path =\n    Gren.Kernel.FileSystem.unlink path\n\n\n-- SUBSCRIPTION\n\n\ntype FileSystemSub msg\n    = Watch Path Bool (WatchEvent -> msg)\n\n\n{-| Represents a change within a watched directory.\n\n* `Changed` means that the contents of a file has changed in some way.\n* `Moved` means that an entity has been added or removed. A rename is usually two `Moved` events.\n\nOn most operating systems, each event will be associated with a [Path](FileSystem.Path#Path)\nrelative to the watched directory, but some operating systems will not provide that information.\n-}\ntype WatchEvent\n    = Changed (Maybe Path)\n    | Moved (Maybe Path)\n\n\nsubMap : (a -> b) -> FileSystemSub a -> FileSystemSub b\nsubMap mapFn sub =\n    case sub of\n        Watch path recursive msgMap ->\n            Watch path recursive (mapFn << msgMap)\n\n\n{-| This notifies your application every time there is a change within the directory\nrepresented by the given [Path](FileSystem.Path#Path).\n-}\nwatch : Permission -> (WatchEvent -> msg) -> Path -> Sub msg\nwatch _ msgMap path =\n    subscription (Watch path False msgMap)\n\n\n{-| Same as [watch](#watch), but this will also watch for changes in sub-directories.\n-}\nwatchRecursive : Permission -> (WatchEvent -> msg) -> Path -> Sub msg\nwatchRecursive _ msgMap path =\n    subscription (Watch path True msgMap)\n\n\n-- LOOP\n\n\ntype alias State msg =\n    { watchers : Dict String (Taggers msg)\n    , recursiveWatchers : Dict String (Taggers msg)\n    , processes : Dict String Process.Id\n    }\n\n\ntype alias Taggers msg =\n    Array (WatchEvent -> msg)\n\n\ninit : Task Never (State msg)\ninit =\n    Task.succeed\n        { watchers = Dict.empty\n        , recursiveWatchers = Dict.empty\n        , processes = Dict.empty\n        }\n\n\nonEffects\n    : Platform.Router msg SelfMsg\n    -> Array (FileSystemSub msg)\n    -> State msg\n    -> Task.Task Never (State msg)\nonEffects router subs state =\n    let\n        newWatchers =\n            subs\n                |> Array.filter (\\(Watch _ bool _) -> not bool)\n                |> Array.foldl subToWatcher Dict.empty\n\n        newRecursiveWatchers =\n            subs\n                |> Array.filter (\\(Watch _ bool _) -> bool)\n                |> Array.foldl subToWatcher Dict.empty\n\n        toSpawn =\n            Dict.diff newWatchers state.watchers\n                |> Dict.keys\n        \n        recursiveToSpawn =\n            Dict.diff newRecursiveWatchers state.recursiveWatchers\n                |> Dict.keys\n        \n        toStop =\n            Dict.diff state.watchers newWatchers\n                |> Dict.keys\n        \n        recursiveToStop =\n            Dict.diff state.recursiveWatchers newRecursiveWatchers\n                |> Dict.keys\n\n        asRecursiveKey key =\n            key ++ \"$recursive\"\n\n        flipFold fn array initial =\n            Array.foldl fn initial array\n\n        taskStopper keyMapper =\n            (\\key accTask ->\n                Task.andThen \n                    (\\processes ->\n                        let\n                            realKey = \n                                keyMapper key\n                        in\n                        case Dict.get realKey processes of\n                            Just processId ->\n                                Process.kill processId\n                                    |> Task.map (\\_ -> Dict.remove realKey processes)\n\n                            Nothing ->\n                                accTask\n                    )\n                    accTask\n            )\n\n        taskSpawner keyMapper isRecursive =\n            (\\key accTask ->\n                accTask\n                    |> Task.andThen \n                        (\\processes ->\n                            Process.spawn (attachWatcher key isRecursive (Platform.sendToSelf router << WatchPathChange isRecursive key))\n                                |> Task.map (\\id -> Dict.set (keyMapper key) id processes)\n                        )\n            )\n    in\n    Array.foldl \n        (taskStopper identity)\n        (Task.succeed state.processes)\n        toStop\n    |> flipFold (taskStopper asRecursiveKey) recursiveToStop\n    |> flipFold (taskSpawner identity False) toSpawn\n    |> flipFold (taskSpawner asRecursiveKey True) recursiveToSpawn\n    |> Task.map\n        (\\newProcesses ->\n            { watchers = newWatchers\n            , recursiveWatchers = newRecursiveWatchers\n            , processes = newProcesses\n            }\n        )\n\n\nsubToWatcher : FileSystemSub msg -> Dict String (Taggers msg) -> Dict String (Taggers msg)\nsubToWatcher sub taggers =\n    case sub of\n        Watch path _ tagger ->\n            let\n                key =\n                    Gren.Kernel.FilePath.toString path\n            in\n            case Dict.get key taggers of\n                Just data ->\n                    Dict.set \n                        key\n                        (Array.pushLast tagger data)\n                        taggers\n\n                Nothing ->\n                    Dict.set key [ tagger ] taggers\n\n\nattachWatcher : String -> Bool -> (WatchEvent -> Task.Task Never {}) -> Task.Task x {}\nattachWatcher =\n  Gren.Kernel.FileSystem.watch\n\n\ntype SelfMsg\n    = WatchPathChange Bool String WatchEvent\n\n\nonSelfMsg : Platform.Router msg SelfMsg -> SelfMsg -> State msg -> Task.Task Never (State msg)\nonSelfMsg router (WatchPathChange recursive key event) state =\n    let\n        notifyApplication taggers =\n            Array.foldl \n                (\\tagger tasks -> \n                    tasks\n                        |> Task.andThen (\\{} -> Platform.sendToApp router (tagger event))\n                )\n                (Task.succeed {})\n                taggers\n                |> Task.map (\\{} -> state)\n    in\n    if recursive then\n        case Dict.get key state.recursiveWatchers of\n            Just taggers ->\n                notifyApplication taggers\n\n            Nothing ->\n                Task.succeed state\n\n    else\n        case Dict.get key state.watchers of\n            Just taggers ->\n                notifyApplication taggers\n\n            Nothing ->\n                Task.succeed state\n\n\n-- SPECIAL PATHS\n\n\n{-| Find the [Path](FileSystem.Path#Path) that represents the home directory of the current user.\n-}\nhomeDirectory : Permission -> Task x Path\nhomeDirectory _ =\n    Gren.Kernel.FileSystem.homeDir\n\n\n{-| Returns the current working directory of the program.\n\nThis is the directory that all relative paths are relative to, and is usually the\ndirectory that the program was executed from.\n-}\ncurrentWorkingDirectory : Permission -> Task x Path\ncurrentWorkingDirectory _ =\n    Gren.Kernel.FileSystem.currentWorkingDirectory\n\n\n{-| Find a [Path](FileSystem.Path#Path) that represents a directory meant to hold temporary files.\n-}\ntmpDirectory : Permission -> Task x Path\ntmpDirectory _ =\n    Gren.Kernel.FileSystem.tmpDir\n\n\n{-| [Path](FileSystem.Path#Path) to a file which is always empty. Anything written to this file will be discarded.\n-}\ndevNull : Permission -> Task x Path\ndevNull _ =\n    Gren.Kernel.FileSystem.devNull\n",
        "module Main exposing (main)\n\nimport Node\nimport ChildProcess\nimport Init\nimport Stream exposing (Stream)\nimport Task exposing (Task)\nimport Dict exposing (Dict)\nimport FileSystem\nimport FileSystem.Path as Path exposing (Path)\nimport HttpClient\nimport Bytes exposing (Bytes)\nimport Terminal\nimport Process\n\n\nmain : Node.Program Model Msg\nmain =\n    Node.defineProgram\n        { init = init\n        , update = update\n        , subscriptions = \\_model -> Sub.none\n        }\n\n\ntype alias Model = \n    { args : Array String\n    , stdout : Stream\n    , stderr : Stream\n    , useColor : Bool\n    , fsPermission : FileSystem.Permission\n    , cpPermission : ChildProcess.Permission\n    , httpPermission : HttpClient.Permission\n    , remotePath : Maybe String\n    , localPath : Path\n    , pathToString : Path -> String\n    }\n\n\ncompilerVersion : String\ncompilerVersion =\n    \"0.4.4\"\n\ncountDown : Int -> {} -> Int\ncountDown n rec =\n    if n <= 0 then\n        0\n\n    else\n        countDown (n - 1) rec\n\ninit : Node.Environment -> Init.Task { model : Model, command : Cmd Msg }\ninit env =\n    Init.await FileSystem.initialize <| \\fsPermission ->\n    Init.await ChildProcess.initialize <| \\cpPermission ->\n    Init.await HttpClient.initialize <| \\httpPermission ->\n    Init.await Terminal.initialize <| \\terminalConfig ->\n    Init.awaitTask Node.getEnvironmentVariables <| \\envVars ->\n    Init.awaitTask (FileSystem.homeDirectory fsPermission) <| \\homeDir ->\n            let\n                userArgs =\n                    Array.dropFirst 2 env.args\n\n                useless = countDown 10 {}\n\n                useColor =\n                    case terminalConfig of\n                        Nothing ->\n                            False\n\n                        Just _ ->\n                            case Dict.get \"NO_COLOR\" envVars of\n                                Just _ ->\n                                    False\n\n                                Nothing ->\n                                    True\n\n                maybePaths =\n                    case { platform = env.platform, arch = env.cpuArchitecture, override = Dict.get \"GREN_BIN\" envVars } of\n                        { override = Just overridePath, platform = Node.Win32 } ->\n                            Just <|\n                                { args = userArgs\n                                , stdout = env.stdout\n                                , remotePath = Nothing\n                                , localPath = Path.fromWin32String overridePath\n                                }\n                        \n                        { override = Just overridePath } ->\n                            Just <|\n                                { args = userArgs\n                                , stdout = env.stdout\n                                , remotePath = Nothing\n                                , localPath = Path.fromPosixString overridePath\n                                }\n                                \n                        { platform = Node.Win32, arch = Node.X64 } ->\n                            Just <|\n                                { args = userArgs\n                                , stdout = env.stdout\n                                , remotePath = Just <| makeRemotePath \"gren.exe\"\n                                , localPath = makeLocalPath env.platform homeDir envVars\n                                }\n\n                        { platform = Node.Darwin } ->\n                            Just <|\n                                { args = userArgs\n                                , stdout = env.stdout\n                                , remotePath = Just <| makeRemotePath \"gren_mac\"\n                                , localPath = makeLocalPath env.platform homeDir envVars\n                                }\n\n                        { platform = Node.Linux, arch = Node.X64 } ->\n                            Just <|\n                                { args = userArgs\n                                , stdout = env.stdout\n                                , remotePath = Just <| makeRemotePath \"gren_linux\"\n                                , localPath = makeLocalPath env.platform homeDir envVars\n                                }\n\n                        _ ->\n                            Nothing\n\n                model =\n                    case maybePaths of\n                        Just paths ->\n                            { args = userArgs\n                            , stdout = env.stdout\n                            , stderr = env.stderr\n                            , useColor = useColor\n                            , fsPermission = fsPermission\n                            , cpPermission = cpPermission\n                            , httpPermission = httpPermission\n                            , remotePath = paths.remotePath\n                            , localPath = paths.localPath\n                            , pathToString =\n                                if env.platform == Node.Win32 then\n                                    Path.toWin32String\n                                else\n                                    Path.toPosixString\n                            }\n                        \n                        Nothing ->\n                            -- dummy model\n                            { args = []\n                            , stdout = env.stdout\n                            , stderr = env.stderr\n                            , useColor = useColor\n                            , fsPermission = fsPermission\n                            , cpPermission = cpPermission\n                            , httpPermission = httpPermission\n                            , remotePath = Nothing\n                            , localPath = Path.empty\n                            , pathToString = Path.toPosixString\n                            }\n            in\n            Node.startProgram\n                { model = model\n                , command =\n                    case maybePaths of\n                        Just _ ->\n                            FileSystem.checkAccess fsPermission [] model.localPath\n                                |> Task.attempt ExistanceChecked\n\n                        Nothing ->\n                            Stream.sendLine env.stderr \"We currently don't support this platform/arch.\"\n                                |> Task.execute\n                }\n\n\nmakeRemotePath : String -> String\nmakeRemotePath filename =\n    String.join \"/\"\n        [ \"https://github.com/gren-lang/compiler/releases/download\"\n        , compilerVersion\n        , filename\n        ]\n\n\nmakeLocalPath : Node.Platform -> Path -> Dict String String -> Path\nmakeLocalPath platform homeDir envVars =\n    let\n        startPath =\n            case platform of\n                Node.Win32 ->\n                    envVars\n                        |> Dict.get \"LOCALAPPDATA\"\n                        |> Maybe.map Path.fromWin32String\n                        |> Maybe.withDefault (\n                            \"AppData/Local\"\n                                |> Path.fromPosixString\n                                |> Path.prepend homeDir\n                            )\n\n                Node.Darwin ->\n                    \"Library/Caches\"\n                        |> Path.fromPosixString\n                        |> Path.prepend homeDir\n\n                _ ->\n                    envVars\n                        |> Dict.get \"XDG_CACHE_HOME\"\n                        |> Maybe.map Path.fromPosixString\n                        |> Maybe.withDefault (Path.append (Path.fromPosixString \".cache\") homeDir)\n\n        filename =\n            case platform of\n                Node.Win32 ->\n                    \"gren.exe\"\n\n                _ ->\n                    \"gren\"\n        \n        endPath =\n            [ \"gren\"\n            , compilerVersion\n            , \"bin\"\n            , filename\n            ]\n                |> String.join \"/\"\n                |> Path.fromPosixString\n    in\n    Path.prepend startPath endPath\n\n\ntype Msg\n    = ExistanceChecked (Result FileSystem.Error Path)\n    | CompilerDownloaded (Result (HttpClient.Error Bytes) (HttpClient.Response Bytes))\n    | CompilerInstalled (Result FileSystem.Error {})\n\n\nupdate : Msg -> Model -> { model : Model, command : Cmd Msg }\nupdate msg model =\n    case msg of\n        ExistanceChecked (Err _) ->\n            { model = model\n            , command =\n                case model.remotePath of\n                    Just remotePath ->\n                        Stream.sendLine model.stdout (\"Compiler not found at \" ++ model.pathToString model.localPath ++ \". Downloading...\")\n                            |> Task.andThen (\\{} -> downloadBinary model.httpPermission remotePath)\n                            |> Task.attempt CompilerDownloaded\n                    \n                    Nothing ->\n                        Stream.sendLine model.stderr (\"Compiler not found at \" ++ model.pathToString model.localPath)\n                            |> Task.execute\n            }\n\n        ExistanceChecked (Ok _) ->\n            { model = model\n            , command =\n                runCompiler model\n                    |> Task.execute\n            }\n\n        CompilerDownloaded (Err ((HttpClient.BadStatus res) as err)) ->\n            if res.statusCode == 302 then\n                case Dict.get \"location\" res.headers of\n                    Just [ location ] ->\n                        { model = model\n                        , command =\n                            downloadBinary model.httpPermission location\n                                |> Task.attempt CompilerDownloaded\n                        }\n\n                    _ ->\n                        { model = model\n                        , command =\n                            Stream.sendLine model.stderr \"Missing, or vague, 'location' header in 302 response from server.\"\n                                |> Task.execute\n                        }\n\n            else\n                { model = model\n                , command =\n                    Stream.sendLine model.stderr (HttpClient.errorToString err)\n                        |> Task.execute\n                }\n        \n        CompilerDownloaded (Err err) ->\n                { model = model\n                , command =\n                    Stream.sendLine model.stderr (HttpClient.errorToString err)\n                        |> Task.execute\n                }\n        \n        CompilerDownloaded (Ok res) ->\n            let\n                cacheFolder =\n                    Path.parentPath model.localPath\n                        |> Maybe.withDefault Path.empty\n            in\n            { model = model\n            , command =\n                FileSystem.makeDirectory model.fsPermission { recursive = True } cacheFolder\n                    |> Task.andThen (\\_cacheFolder -> FileSystem.writeFile model.fsPermission res.data model.localPath)\n                    |> Task.andThen \n                        (FileSystem.changeAccess\n                            model.fsPermission\n                            { owner = [ FileSystem.Read, FileSystem.Write, FileSystem.Execute ]\n                            , group = [ FileSystem.Read, FileSystem.Execute ]\n                            , others = [ FileSystem.Read, FileSystem.Execute ]\n                            }\n                        )\n                    |> Task.andThen (\\_binPath -> Stream.sendLine model.stdout \"Downloaded\")\n                    |> Task.attempt CompilerInstalled\n            }\n        \n        CompilerInstalled (Err fsErr) ->\n            { model = model\n            , command =\n                Stream.sendLine model.stderr (\"Failed to install binary after download, due to error: \" ++ FileSystem.errorToString fsErr)\n                    |> Task.execute\n            }\n        \n        CompilerInstalled (Ok {}) ->\n            { model = model\n            , command =\n                runCompiler model\n                    |> Task.execute\n            }\n\n\ndownloadBinary : HttpClient.Permission -> String -> Task (HttpClient.Error Bytes) (HttpClient.Response Bytes)\ndownloadBinary permission url =\n    HttpClient.get url\n        |> HttpClient.expectBytes\n        |> HttpClient.send permission\n\n\nrunCompiler : Model -> Task x Process.Id\nrunCompiler model =\n    let\n        colorEnvVar =\n            if model.useColor then\n                Dict.singleton \"FORCE_COLOR\" \"1\"\n            else\n                Dict.singleton \"NO_COLOR\" \"1\"\n      in\n    ChildProcess.spawn model.cpPermission (model.pathToString model.localPath) model.args <|\n        { ChildProcess.defaultSpawnOptions\n            | environmentVariables = \n                ChildProcess.MergeWithEnvironmentVariables colorEnvVar\n        }\n",
        "module FileSystem.Path exposing \n    ( Path\n    --\n    , empty\n    , fromPosixString\n    , toPosixString\n    , fromWin32String\n    , toWin32String\n    --\n    , filenameWithExtension\n    , parentPath\n    --\n    , append\n    , prepend\n    , join\n    )\n\n\n{-| A path represents the location of a file or directory in a filesystem.\n\n@docs Path\n\n## Constructors\n\n@docs empty, fromPosixString, toPosixString, fromWin32String, toWin32String\n\n## Query\n\n@docs filenameWithExtension, parentPath\n\n## Manipulation\n\n@docs append, prepend, join\n-}\n\n\nimport Task exposing (Task)\nimport Gren.Kernel.FilePath\n\n\n{-| A cross-platform representation of a filesystem path.\n\nIf `root` is empty, it means that the path is relative to the working directory.\nOn posix-compatible systems (Linux, Mac...), the root value is \"/\" if not empty.\nOn Windows, the root refers to the specific disk that the path applies to.\n\n`filename` (and `extension`) refers to the last part of a path. It can still\nrepresent a directory.\n\n-}\ntype alias Path =\n    { root : String\n    , directory : Array String\n    , filename : String\n    , extension : String\n    }\n\n\n{-| The empty [Path](#Path). Normally treated as the current directory.\n-}\nempty : Path\nempty =\n    { root = \"\"\n    , directory = []\n    , filename = \"\"\n    , extension = \"\"\n    }\n\n\n{-| Build a [Path](#Path) from a `String`. The `String` should represent a Posix-compatible path.\n-}\nfromPosixString : String -> Path\nfromPosixString =\n    Gren.Kernel.FilePath.fromPosix\n\n\n{-| String representation of a [Path](#Path) for Posix systems.\n-}\ntoPosixString : Path -> String\ntoPosixString =\n    Gren.Kernel.FilePath.toPosix\n\n\n{-| Build a [Path](#Path) from a `String`. The `String` should represent a Windows-compatible path.\n-}\nfromWin32String : String -> Path\nfromWin32String =\n    Gren.Kernel.FilePath.fromWin32\n\n\n{-| `String` representation of a [Path](#Path) for Windows.\n-}\ntoWin32String : Path -> String\ntoWin32String =\n    Gren.Kernel.FilePath.toWin32\n\n\n{-| Return the filename and file extension for a [Path](#Path).\n\n    \"/home/me/file.md\"\n        |> fromPosixString\n        |> filenameWithExtension\n        -- returns \"file.md\"\n-}\nfilenameWithExtension : Path -> String\nfilenameWithExtension path =\n    if String.isEmpty path.extension then\n        path.filename\n\n    else\n        path.filename ++ \".\" ++ path.extension\n\n\n{-| Return a [Path](#Path) that represents the directory which holds the given [Path](#Path)\n\n    \"/home/me/file.md\"\n        |> fromPosixString\n        |> parentPath\n        -- returns (Just \"/home/me\")\n-}\nparentPath : Path -> Maybe Path\nparentPath path =\n    case Array.popLast path.directory of\n        Nothing ->\n            if filenameWithExtension path == \"\" then\n                Nothing\n\n            else\n                Just\n                    { path\n                        | filename = \"\"\n                        , extension = \"\"\n                    }\n\n        Just { last, initial } ->\n            let\n                { filename, extension } =\n                    case String.split \".\" last of\n                        [ file, ext ] ->\n                            { filename = file\n                            , extension = ext\n                            }\n\n                        _ ->\n                            { filename = last\n                            , extension = \"\"\n                            }\n            in\n            Just\n                { path\n                    | directory = initial\n                    , filename = filename\n                    , extension = extension\n                }\n\n\n{-| Join two paths by appending the first [Path](#Path) onto the second.\n-}\nappend : Path -> Path -> Path\nappend left right =\n    prepend right left\n\n\n{-| Join two paths by prepending the first [Path](#Path) onto the second.\n-}\nprepend : Path -> Path -> Path\nprepend left right =\n    { left\n        | directory =\n            left.directory\n                |> Array.pushLast (filenameWithExtension left)\n                |> Array.append right.directory\n                |> Array.filter (\\dir -> dir /= \"\")\n        , filename = right.filename\n        , extension = right.extension\n    }\n\n\n{-| Join all paths in an `Array`.\n-}\njoin : Array Path -> Path\njoin paths =\n    Array.foldl append (fromPosixString \".\") paths\n",
        "module ChildProcess exposing \n    ( Permission\n    , initialize\n    --\n    , RunOptions\n    , defaultRunOptions\n    , Shell(..)\n    , WorkingDirectory(..)\n    , EnvironmentVariables(..)\n    , RunDuration(..)\n    --\n    , FailedRun\n    , SuccessfulRun\n    , run\n    , runWithDefaultOptions\n    --\n    , SpawnOptions\n    , Connection(..)\n    , defaultSpawnOptions\n    , spawn\n    , spawnWithDefaultOptions\n    )\n\n\n{-| A running program is a process. A process spawned from another process is known as a child process.\n\nThis module allow you to spawn child processes.\n\n## Initialization\n\n@docs Permission, initialize\n\n## Running processes\n\n@docs RunOptions, defaultRunOptions, Shell, WorkingDirectory, EnvironmentVariables, RunDuration\n@docs FailedRun, SuccessfulRun, run, runWithDefaultOptions\n\n## Spawning processes\n\n@docs SpawnOptions, Connection, defaultSpawnOptions, spawn, spawnWithDefaultOptions\n-}\n\n\nimport Gren.Kernel.ChildProcess\nimport Bytes exposing (Bytes)\nimport Dict exposing (Dict)\nimport Task exposing (Task)\nimport Init\nimport Internal.Init\nimport Process\n\n\n{-| This value represents the permission to spawn child processes.\n\nOnly code you trust should have access to this value.\n-}\ntype Permission\n    = Permission\n\n\n{-| Initialize the `ChildProcess` subsystem, which gains you the permission to\nspawn child processes.\n-}\ninitialize : Init.Task Permission\ninitialize =\n    Task.succeed Permission\n        |> Internal.Init.Task\n\n\n-- OPTIONS\n\n\n{-| Options to customize the execution of a child process created with [run](#run).\n\n* `shell` is the shell to run the process in (if any)\n* `workingDirectory` specifies the working directory of the process\n* `environmentVariables` specifies the environment variables the process has access to\n* `maximumBytesWrittenToStreams` specifies an upper bound of bytes that can be returned from the process\n* `runDuration` specifies a maximum amount of time a process is allowed to run before exiting\n-}\ntype alias RunOptions =\n    { shell : Shell\n    , workingDirectory : WorkingDirectory\n    , environmentVariables : EnvironmentVariables\n    , maximumBytesWrittenToStreams : Int\n    , runDuration : RunDuration\n    }\n\n\n{-| A nice default set of options for the [run](#run) function\n-}\ndefaultRunOptions : RunOptions\ndefaultRunOptions =\n    { shell = DefaultShell\n    , workingDirectory = InheritWorkingDirectory\n    , environmentVariables = InheritEnvironmentVariables\n    , maximumBytesWrittenToStreams = 1024 * 1024 -- 1Mb\n    , runDuration = NoLimit\n    }\n\n\n{-| Which shell should the child process run in?\n\n* `NoShell` executes the process directly, without any shell. A little bit more efficient, but you lose some convinience as shell behaviour (like glob patterns) isn't available for arguments\n* `DefaultShell` executes the process in the default shell for the currently running system\n* `CustomShell` executes the process in the specified shell.\n-}\ntype Shell\n    = NoShell\n    | DefaultShell\n    | CustomShell String\n\n\n{-| What should be the working directory of the process?\n\n* `InheritWorkingDirectory` inherits the working directory from its parent\n* `SetWorkingDirectory` sets the working directory to the specified value (doesn't affect parent)\n-}\ntype WorkingDirectory\n    = InheritWorkingDirectory\n    | SetWorkingDirectory String\n\n\n{-| What should be the environment variables of the process?\n\n* `InheritEnvironmentVariables` inherits the environment variables from its parent\n* `MergeWithEnvironmentVariables` inherits the environment variables from its parent, with the specified modifications\n* `ReplaceEnvironmentVariables` sets the environment variables to the specified dictionary\n-}\ntype EnvironmentVariables\n    = InheritEnvironmentVariables\n    | MergeWithEnvironmentVariables (Dict String String)\n    | ReplaceEnvironmentVariables (Dict String String)\n\n\n{-| How long is the process allowed to run before it's forcefully terminated?\n\n* `NoLimit` means it can run forever\n* `Milliseconds` sets the limit to the specified number of milliseconds\n-}\ntype RunDuration\n    = NoLimit\n    | Milliseconds Int\n\n\n-- RUN\n\n\n{-| Return value when a process terminates due to an error\n\nThe exit code provides some hint of what went wrong, but what it means depends on the program which was run.\n-}\ntype alias FailedRun =\n    { exitCode: Int\n    , stdout : Bytes\n    , stderr : Bytes\n    }\n\n\n{-| Return value when a process terminates without error\n-}\ntype alias SuccessfulRun =\n    { stdout : Bytes\n    , stderr : Bytes\n    }\n\n\n{-| Execute a process with a given name, arguments and options, and wait for it to terminate.\n\n    run permission \"cat\" [ \"my_file\" ] defaultRunOptions\n\n-}\nrun : Permission -> String -> Array String -> RunOptions -> Task FailedRun SuccessfulRun\nrun _ program arguments opts =\n    Gren.Kernel.ChildProcess.run\n        { program = program\n        , arguments = arguments\n        , shell =\n            case opts.shell of\n                NoShell ->\n                    { choice = 0\n                    , value = \"\"\n                    }\n\n                DefaultShell ->\n                    { choice = 1\n                    , value = \"\"\n                    }\n\n                CustomShell value ->\n                    { choice = 2\n                    , value = value\n                    }\n        , workingDirectory =\n             case opts.workingDirectory of\n                 InheritWorkingDirectory -> \n                    { inherit = True\n                    , override = \"\"\n                    }\n\n                 SetWorkingDirectory value ->\n                    { inherit = False\n                    , override = value\n                    }\n        , environmentVariables =\n            case opts.environmentVariables of\n                InheritEnvironmentVariables -> \n                    { option = 0\n                    , value = Dict.empty\n                    }\n                \n                MergeWithEnvironmentVariables value ->\n                    { option = 1\n                    , value = value\n                    }\n\n                ReplaceEnvironmentVariables value ->\n                    { option = 2\n                    , value = value\n                    }\n        , maximumBytesWrittenToStreams = opts.maximumBytesWrittenToStreams\n        , runDuration =\n            case opts.runDuration of\n                NoLimit ->\n                    0\n\n                Milliseconds ms -> \n                    max 0 ms\n        }\n\n\n{-| Same as [run](#run), but with [defaultRunOptions](#defaultRunOptions) passed in as options.\n-}\nrunWithDefaultOptions : Permission -> String -> Array String -> Task FailedRun SuccessfulRun\nrunWithDefaultOptions permission program arguments =\n    run permission program arguments defaultRunOptions\n\n\n-- SPAWN\n\n\n{-| Options to customize the execution of a child process created with [spawn](#spawn).\n\n* `shell` is the shell to run the process in (if any)\n* `workingDirectory` specifies the working directory of the process\n* `environmentVariables` specifies the environment variables the process has access to\n* `runDuration` specifies a maximum amount of time a process is allowed to run before exiting\n* `connection` let's you specify how close the new process is connected to the application\n-}\ntype alias SpawnOptions =\n    { shell : Shell\n    , workingDirectory : WorkingDirectory\n    , environmentVariables : EnvironmentVariables\n    , runDuration : RunDuration\n    , connection : Connection\n    }\n\n\n{-| What relation should the newly spawned process have with the running application?\n\n* `Integrated` means that the spawned process shares the stdin, stdout and stderr streams and that the application will wait for its termination.\n* `Ignored` means that stdin, stdout and stderr is seperate but that the application will still wait for its termination.\n* `Detached` means that the application can terminate even if the spawned process is still running.\n-}\ntype Connection\n    = Integrated\n    | Ignored\n    | Detached\n\n\n{-| A nice default set of options for the [spawn](#spawn) function.\n-}\ndefaultSpawnOptions : SpawnOptions\ndefaultSpawnOptions =\n    { shell = DefaultShell\n    , workingDirectory = InheritWorkingDirectory\n    , environmentVariables = InheritEnvironmentVariables\n    , runDuration = NoLimit\n    , connection = Integrated\n    }\n\n\n{-| Spawn a process with a given name, arguments and options, and let it run in the background.\nThis is mostly helpful for starting long-running processes.\n\n    spawn permission \"tail\" [ \"my_file\" ] defaultSpawnOptions\n\n-}\nspawn : Permission -> String -> Array String -> SpawnOptions -> Task x Process.Id\nspawn _ program arguments opts =\n    Process.spawn <|\n        Gren.Kernel.ChildProcess.spawn\n            { program = program\n            , arguments = arguments\n            , shell =\n                case opts.shell of\n                    NoShell ->\n                        { choice = 0\n                        , value = \"\"\n                        }\n\n                    DefaultShell ->\n                        { choice = 1\n                        , value = \"\"\n                        }\n\n                    CustomShell value ->\n                        { choice = 2\n                        , value = value\n                        }\n            , workingDirectory =\n                 case opts.workingDirectory of\n                     InheritWorkingDirectory -> \n                        { inherit = True\n                        , override = \"\"\n                        }\n\n                     SetWorkingDirectory value ->\n                        { inherit = False\n                        , override = value\n                        }\n            , environmentVariables =\n                case opts.environmentVariables of\n                    InheritEnvironmentVariables -> \n                        { option = 0\n                        , value = Dict.empty\n                        }\n                \n                    MergeWithEnvironmentVariables value ->\n                        { option = 1\n                        , value = value\n                        }\n\n                    ReplaceEnvironmentVariables value ->\n                        { option = 2\n                        , value = value\n                        }\n            , runDuration =\n                case opts.runDuration of\n                    NoLimit ->\n                        0\n\n                    Milliseconds ms -> \n                        max 0 ms\n            , connection =\n                case opts.connection of\n                    Integrated ->\n                        0\n\n                    Ignored ->\n                        1\n\n                    Detached ->\n                        2\n            }\n\n\n{-| Same as [spawn], but with [defaultSpawnOptions](#defaultSpawnOptions) passed in as options.\n-}\nspawnWithDefaultOptions : Permission -> String -> Array String -> Task x Process.Id\nspawnWithDefaultOptions permission program arguments =\n    spawn permission program arguments defaultSpawnOptions\n",
        "effect module HttpClient where { command = MyCmd } exposing\n    ( Permission\n    , initialize, initializeForHost\n    , RequestConfiguration, get, post, request\n    , defaultTimeout, withTimeout\n    , withHeader, withDuplicatedHeader\n    , Body, withEmptyBody, withStringBody, withJsonBody, withBytesBody\n    , Expect, expectAnything, expectNothing, expectString, expectJson, expectBytes\n    , send\n    , Response\n    , Error(..), errorToString\n    , StreamRequest, StreamEvent(..), stream, sendChunk, startReceive, abort\n    )\n\n\n{-|\n\nA module for communicating over HTTP.\n\nYou start by building a [RequestConfiguration](#RequestConfiguration) type, which represents the request you'll make to a server. Once done,\nyou can either do a `send`, which represents the response as a `Task`, or `stream` which will allow you to perform\nactions while the request is sending and while the response is coming in. A typical example of why you'd use `stream`\nis to show a progress bar to the user, or decode the response incrementally as opposed to all at once.\n\nSee [examples/http-client](https://github.com/gren-lang/node/tree/main/examples/http-client) for a working example.\n\n## Initialization\n\nCode that wishes to perform HTTP requests require a permission to do so.\n\n@docs Permission, initialize, initializeForHost\n\n## Request configuration\n\nIn order to send something over HTTP, you first need a description of how that request will look like.\n\n@docs RequestConfiguration, get, post, request\n\n## Timeouts\n\nA timeout represents how long you're willing to wait before giving up on receiving\na response from the server. Servers might not respond for any number of reasons, like bugs or huge amounts of traffic,\nso it is a good idea to return an error to the user instead of waiting \"forever\" for a response.\n\n@docs defaultTimeout, withTimeout\n\n## Headers\n\nEvery HTTP request can have arbitrary metadata attached, called headers. Headers allow you to attach things like\nauthorization information, how the body is encoded or the name of the client making the request.\n\nIt might be interesting to read this [list of HTTP header fields](https://en.wikipedia.org/wiki/List_of_HTTP_header_fields).\n\n@docs withHeader, withDuplicatedHeader\n\n## Request body\n\nThe request body is the actual data that you wish to send to a server.\n\n@docs Body, withEmptyBody, withStringBody, withJsonBody, withBytesBody\n\n## Expected response body\n\nOnce a request has been sent, you usually get a response. The `Expect` type represents\nwhat we expect the response body to be.\n\n@docs Expect, expectAnything, expectNothing, expectString, expectJson, expectBytes\n\n## Send\n\nOnce your `Response` is configured, you'll want to actually send the request.\n\n@docs send\n\n@docs Response\n\n## Errors\n\n@docs Error, errorToString\n\n## Streaming\n\nStreaming is the more advanced way to perform a HTTP request. This requires that you follow the Elm\narchitecture, as you'll receive messages for every chunk of data sent and received. The benefit of this\nextra complexity, is that you can perform actions while the request is being performed.\n\n@docs StreamRequest, StreamEvent, stream, sendChunk, startReceive, abort\n\n-}\n\n\nimport Dict exposing (Dict)\nimport Init\nimport Internal.Init\nimport Json.Encode as Json\nimport Json.Decode\nimport Bytes exposing (Bytes)\nimport Task exposing (Task)\nimport Platform\nimport Gren.Kernel.HttpClient\nimport HttpServer exposing (Method(..), methodToString)\n\n\n{-| A value that represents the permission to perform HTTP requests.\n\nOnly code you trust should be granted permission.\n-}\ntype Permission\n    = AnyPermission\n    | SpecificPermission String\n\n\n{-| Call this during application initialization to get the permission to perform any kind of HTTP request.\n-}\ninitialize : Init.Task Permission\ninitialize =\n    Task.succeed AnyPermission\n        |> Internal.Init.Task\n\n\n{-| Call during application initialization to get a host-specific permission. Code that has this permission value, will only\nbe able to contact a specific host.\n-}\ninitializeForHost : String -> Init.Task Permission\ninitializeForHost host =\n    Task.succeed (SpecificPermission host)\n        |> Internal.Init.Task\n\n\n-- REQUEST CONFIGURATION\n\n\n{-| Describes the request to be made. Use [get](#get), [post](#post) or [request](#request) to initialize\nthis value, then customize it using the following `with` functions.\n-}\ntype alias RequestConfiguration responseType =\n    { method : Method\n    , url : String\n    , headers : Dict String (Array String)\n    , body : Body\n    , expect : Expect responseType\n    , timeout : Int\n    }\n\n\n{-| Initializes the configuration for a simple GET request to the given url.\n-}\nget : String -> RequestConfiguration {}\nget url =\n    request GET url\n\n\n{-| Initializes the configuration for a simple POST request to the given url.\n-}\npost : String -> RequestConfiguration {}\npost url =\n    request POST url\n\n\n{-| Initializes a request configuration with the given method and url.\n-}\nrequest : Method -> String -> RequestConfiguration {}\nrequest method url =\n    { method = method\n    , url = url\n    , headers = Dict.empty\n    , body = BodyEmpty\n    , expect = ExpectAnything\n    , timeout = defaultTimeout\n    }\n\n\n{-| This is the default timeout value. It is set to 10 seconds.\nIf you don't use [withTimeout](#withTimeout) to set a timeout specifically, \nthis value will be used.\n-}\ndefaultTimeout : Int\ndefaultTimeout =\n    -- 10 seconds\n    10 * 1000\n\n\n{-| Lets you specify a timeout, in milliseconds, for a request.\nIf the server doesn't respond to your request within the given timeout, the request\nwill fail with a Timeout [Error](#Error).\n-}\nwithTimeout : Int -> RequestConfiguration a -> RequestConfiguration a\nwithTimeout ms req =\n    if ms < 0 then\n        req\n\n    else\n        { req | timeout = ms }\n\n\n{-| A header is a key-value pair of strings that says something about the request.\nExamples include the length of the body, authentication information, name of the client making the request, etc.\n-}\nwithHeader : String -> String -> RequestConfiguration a -> RequestConfiguration a\nwithHeader key value req =\n    { req\n        | headers = Dict.set (String.toLower key) [value] req.headers\n    }\n\n\n{-| Header keys doesn't have to be unique. You're allowed to send the same kind of header\nmultiple times, like sending multiple cookies. The behaviour of [withHeader](#withHeader) will\nreplace the value of an already set header. This function will not.\n-}\nwithDuplicatedHeader : String -> String -> RequestConfiguration a -> RequestConfiguration a\nwithDuplicatedHeader key value req =\n    { req\n        | headers = Dict.update \n            (String.toLower key)\n            (Maybe.map (Array.pushLast value) >> Maybe.withDefault [value] >> Just)\n            req.headers\n    }\n\n\n-- BODY\n\n\n{-| The body represents the main data that you will send in the HTTP request.\n-}\ntype Body\n    = BodyEmpty\n    | BodyString String\n    | BodyBytes Bytes\n\n\nbodyTypeAsString : Body -> String\nbodyTypeAsString body =\n    case body of\n        BodyEmpty ->\n            \"EMPTY\"\n\n        BodyString _ ->\n            \"STRING\"\n\n        BodyBytes _ ->\n            \"BYTES\"\n\n\n{-| Removes the body from the [RequestConfiguration](#RequestConfiguration).\nYou normally don't have to use this function, as an empty body is the default.\n\nIf the \"Content-Type\" header is set, this function will remove it.\n-}\nwithEmptyBody : RequestConfiguration a -> RequestConfiguration a\nwithEmptyBody req = \n    { req\n        | headers = Dict.remove \"content-type\" req.headers\n        , body = BodyEmpty\n    }\n\n\n{-| Sets the given string as the request body. You need to provide a mime type to\ndescribe what the string contains. This mime type will be set as the \"Content-Type\"\nheader, potentially overwriting the header if it has already been set.\n-}\nwithStringBody : String -> String -> RequestConfiguration a -> RequestConfiguration a\nwithStringBody mimeType value req = \n    { req\n        | headers = Dict.set \"content-type\" [mimeType] req.headers\n        , body = BodyString value\n    }\n\n\n{-| Sets the provided Json value the request body. A \"Content-Type\" header will be\nattached to the request with a value of \"application/json\", potentially overwriting\nthe header if it has already been set.\n-}\nwithJsonBody : Json.Value -> RequestConfiguration a -> RequestConfiguration a\nwithJsonBody value req = \n    withStringBody \"application/json\" (Json.encode 0 value) req\n\n\n{-| Sets the provided Bytes value as the request body. You need to provide a mime type to\ndesribe what the bytes represent. This mime type will be set as the \"Content-Type\" header,\npotentially overwriting the header if it has already been set.\n-}\nwithBytesBody : String -> Bytes -> RequestConfiguration a -> RequestConfiguration a\nwithBytesBody mimeType value req = \n    { req\n        | headers = Dict.set \"content-type\" [mimeType] req.headers\n        , body = BodyBytes value\n    }\n\n\n-- EXPECT\n\n\n{-| This describes what you expect the server will respond with when it receives your request.\n-}\ntype Expect a\n    = ExpectNothing\n    | ExpectAnything\n    | ExpectString\n    | ExpectJson (Json.Decode.Decoder a)\n    | ExpectBytes\n\n\nexpectTypeAsString : Expect a -> String\nexpectTypeAsString expect =\n    case expect of\n        ExpectNothing ->\n            \"NOTHING\"\n\n        ExpectAnything ->\n            \"ANYTHING\"\n\n        ExpectString ->\n            \"STRING\"\n\n        ExpectJson _ ->\n            \"JSON\"\n\n        ExpectBytes ->\n            \"BYTES\"\n\n\n{-| Use this when you you don't really care what the server responds with. Anything is fine.\nActually, this is the default value so you probably don't need to use this at all.\n-}\nexpectAnything : RequestConfiguration a -> RequestConfiguration {}\nexpectAnything req =\n    -- Need to create a new record for type checking to pass\n    { method = req.method\n    , url = req.url\n    , headers = req.headers\n    , body = req.body\n    , expect = ExpectAnything\n    , timeout = req.timeout\n    }\n\n\n{-| Expect _exactly_ nothing. Use this when you want a request to fail if the server responds with\nanything at all.\n-}\nexpectNothing : RequestConfiguration a -> RequestConfiguration {}\nexpectNothing req =\n    { method = req.method\n    , url = req.url\n    , headers = req.headers\n    , body = req.body\n    , expect = ExpectNothing\n    , timeout = req.timeout\n    }\n\n\n{-| Use this when you expect the server to respond with a string.\n-}\nexpectString : RequestConfiguration a -> RequestConfiguration String\nexpectString req =\n    { method = req.method\n    , url = req.url\n    , headers = req.headers\n    , body = req.body\n    , expect = ExpectString\n    , timeout = req.timeout\n    }\n\n\n{-| Use this when you expect a Json response. The request will fail if the provided decoder fails.\n-}\nexpectJson : Json.Decode.Decoder a -> RequestConfiguration x -> RequestConfiguration a\nexpectJson decoder req = \n    { method = req.method\n    , url = req.url\n    , headers = req.headers\n    , body = req.body\n    , expect = ExpectJson decoder\n    , timeout = req.timeout\n    }\n\n\n{-| Use this when you want to treat the response as bytes. This will likely never fail, as anything\ncan be interpreted as bytes.\n-}\nexpectBytes : RequestConfiguration a -> RequestConfiguration Bytes\nexpectBytes req =\n    { method = req.method\n    , url = req.url\n    , headers = req.headers\n    , body = req.body\n    , expect = ExpectBytes\n    , timeout = req.timeout\n    }\n\n\n-- SIMPLE SEND\n\n\n{-| Send a request. The task will either complete with a successful [Response](#Response), or an [Error](#Error).\n-}\nsend\n    : Permission \n    -> RequestConfiguration expectedBody\n    -> Task (Error expectedBody) (Response expectedBody)\nsend permission config =\n    Gren.Kernel.HttpClient.request (kernelRequestConfig permission config)\n\n\ntype alias KernelRequestConfig a =\n    { method : String\n    , url : String\n    , headers : Dict String (Array String)\n    , bodyType : String\n    , body : Body\n    , expectType : String\n    , expect : Expect a\n    , timeout : Int\n    }\n\n\nkernelRequestConfig : Permission -> RequestConfiguration a -> KernelRequestConfig a\nkernelRequestConfig permission config =\n    let\n        actualUrl =\n            case permission of\n                AnyPermission ->\n                    config.url\n\n                SpecificPermission prefix ->\n                    prefix ++ config.url\n    in\n    { method = methodToString config.method\n    , url = actualUrl\n    , headers = config.headers\n    , bodyType = bodyTypeAsString config.body\n    , body = config.body\n    , expectType = expectTypeAsString config.expect\n    , expect = config.expect\n    , timeout = config.timeout\n    }\n\n\n-- RESPONSE\n\n\n{-| The response from the server.\n\n* statusCode: A numerical value that gives an indication of how the request went.\nIt might be a good idea to read this [list of HTTP status codes](https://en.wikipedia.org/wiki/List_of_HTTP_status_codes).\n* statusText: A human readable interpretation of the status code.\n* headers: The headers returned by the server.\n* data: The data returned by the server. The type depends on the [Expect](#Expect) value you set on the request.\n-}\ntype alias Response data =\n    { statusCode : Int\n    , statusText : String\n    , headers : Dict String (Array String)\n    , data : data\n    }\n\n\n-- ERRORS\n\n\n{-| A HTTP request can fail in a number of ways.\n\n* BadUrl: Something is wrong with the URL you provided.\n* BadHeaders: The request headers are invalid. Make sure you only use characters in the latin-1 character set.\n* BadStatus: The status code indicates that the response didn't go well. The [Response](#Response) is attached, with a string-encoded data.\n* Timeout: The request timed out. The server didn't respond as quickly as you expected it would.\n* UnknownError: We don't know what went wrong. You should probably report this if you see it in the wild.\n-}\ntype Error body\n    = BadUrl String\n    | BadHeaders\n    | BadStatus (Response body)\n    | UnexpectedResponseBody String\n    | Timeout\n    | UnknownError String\n\n\n{-| Gives a brief description of an error.\n-}\nerrorToString : Error body -> String\nerrorToString err =\n    case err of\n        Timeout ->\n            \"Timeout\"\n\n        BadUrl url ->\n            \"Bad URL: \" ++ url\n\n        BadHeaders ->\n            \"Bad headers: one or more of your headers contains invalid characters.\"\n\n        BadStatus res ->\n            \"Bad status: \" ++ String.fromInt res.statusCode ++ \" - \" ++ res.statusText\n\n        UnexpectedResponseBody message ->\n            \"Unexpected response body: \" ++ message\n\n        UnknownError debugStr ->\n            \"Unknown error: \" ++ debugStr\n\n\n-- STREAM\n\n\n{-| Identifies a streaming request. Required to perform certain operations while\nthe request is streaming.\n-}\ntype StreamRequest =\n    StreamRequest Int\n\n\n{-| When a request is streaming, the application is notified of important events\nand is required to act on those events for the request to be successful.\n\n* SentChunk: The initial request body, or the last piece of data sent with [sendChunk](#sendChunk) has been sent.\nSend more data, or call `startReceive` to begin listening for the response.\n* ReceivedChunk: The server has responded with some data. More data might be coming in, though.\nThe `Done` event will be triggered when there's no more data coming. You can use the provided\n`Response` object to access the response headers, and decide if you'd like to [abort](#abort) the\nrequest or not.\n* Error: Something went wrong. More information in the provided [Error](#Error) object.\n* Aborted: You called [abort](#abort) on this request.\n* Done: The server has sent all it's going to send. You're done.\n-}\ntype StreamEvent\n    = SentChunk StreamRequest\n    | ReceivedChunk StreamRequest (Response Bytes)\n    | Error (Error Bytes)\n    | Aborted\n    | Done\n\n\n{-| Initialize a streaming request. You need to provide a function that generates a message\nfor handling [StreamEvent](#StreamEvent)s. The headers and data will be sent to the server\nimmedietly, and a `SentChunk` event will be sent they're done.\n\nTo tell different requests apart, you can use a partially applied custom type like this:\n\n    type Msg = HttpRequest String StreamEvent\n\n    HttpClient.stream httpPermission (HttpRequest \"Request 1\") requestConfig\n-}\nstream : Permission -> (StreamEvent -> msg) -> RequestConfiguration Bytes -> Cmd msg\nstream permission toMsg config =\n    command <| Start toMsg <| kernelRequestConfig permission config\n\n\n{-| Send more data to the server. This allows you to generate more data as you need to, enabling\nyou slice up a potentially costly, memory heavy or long-running operation over time.\n\nYou don't have to wait for the matching `SentChunk` event before sending more data but keep in\nmind that data will be kept in memory until sent, potentially causing out-of-memory errors in\nthe case of large amounts of data.\n\nIf you're already receiving data from the server, calling this function will no effect.\n-}\nsendChunk : StreamRequest -> Bytes -> Cmd msg\nsendChunk req bytes =\n    command <| SendChunk bytes req\n\n\n{-| Use this when you're done sending data. The server will now begin streaming you the response.\n-}\nstartReceive : StreamRequest -> Cmd msg\nstartReceive req =\n    command <| StartReceive req\n\n\n{-| Stops the request, for any reason, at any time. Useful if you have an unexpected error with\nyour own source of data, or if the server response is one you know you don't want to handle after\nhaving inspected the headers.\n-}\nabort : StreamRequest -> Cmd a\nabort req =\n    command <| Abort req\n\n\n-- COMMANDS\n\n\ntype MyCmd msg\n    = Start (StreamEvent -> msg) (KernelRequestConfig Bytes)\n    | SendChunk Bytes StreamRequest\n    | StartReceive StreamRequest\n    | Abort StreamRequest\n\n\ncmdMap : (a -> b) -> MyCmd a -> MyCmd b\ncmdMap func cmd =\n    case cmd of\n        Start toMsg req ->\n            Start (toMsg >> func) req\n\n        SendChunk bytes req ->\n            SendChunk bytes req\n        \n        StartReceive req ->\n            StartReceive req\n\n        Abort req ->\n            Abort req\n\n\ntype alias RequestsState msg =\n    { nextId : Int \n    , msgHandlers : Dict Int { toMsg : (StreamEvent -> msg), kernelRequest : KernelRequest }\n    }\n\n\ntype KernelRequest = \n    -- Actual implementation provided by kernel code\n    KernelRequest\n\n\ninit : Task Never (RequestsState msg)\ninit =\n  Task.succeed\n    { nextId = 0\n    , msgHandlers = Dict.empty\n    }\n\n\nonEffects : Platform.Router msg SelfMsg -> Array (MyCmd msg) -> RequestsState msg -> Task Never (RequestsState msg)\nonEffects router commands state =\n  case Array.popFirst commands of\n    Nothing ->\n      Task.succeed state\n\n    Just { first, rest } ->\n        case first of\n            Start toMsg config ->\n                let\n                    streamingReq =\n                        StreamRequest state.nextId\n                in\n                Gren.Kernel.HttpClient.stream \n                    (Platform.sendToSelf router << Cleanup) \n                    (Platform.sendToApp router << toMsg) \n                    streamingReq \n                    config\n                    |> Task.map \n                        (\\kernelRequest -> \n                            { state \n                                | nextId = state.nextId + 1\n                                , msgHandlers = \n                                    Dict.set \n                                        state.nextId\n                                        { toMsg = toMsg\n                                        , kernelRequest = kernelRequest\n                                        }\n                                        state.msgHandlers\n                            }\n                        )\n                    |> Task.andThen (\\updatedState -> onEffects router rest updatedState)\n\n            SendChunk bytes ((StreamRequest reqId) as req) ->\n                case Dict.get reqId state.msgHandlers of\n                    Just msgHandler ->\n                        Gren.Kernel.HttpClient.sendChunk \n                            (Platform.sendToApp router << msgHandler.toMsg) \n                            msgHandler.kernelRequest\n                            req \n                            bytes\n                            |> Task.andThen (\\_ -> onEffects router rest state)\n                    \n                    Nothing ->\n                        onEffects router rest state\n            \n            StartReceive ((StreamRequest reqId) as req)->\n                case Dict.get reqId state.msgHandlers of\n                    Just msgHandler ->\n                        Gren.Kernel.HttpClient.startReceive\n                            (Platform.sendToSelf router << Cleanup)\n                            (Platform.sendToApp router << msgHandler.toMsg) \n                            msgHandler.kernelRequest\n                            req \n                            |> Task.andThen (\\_ -> onEffects router rest state)\n                    \n                    Nothing ->\n                        onEffects router rest state\n\n            Abort (StreamRequest reqId)->\n                case Dict.get reqId state.msgHandlers of\n                    Just msgHandler ->\n                        Gren.Kernel.HttpClient.abort msgHandler.kernelRequest\n                            |> Task.andThen (\\_ -> onEffects router rest state)\n                    \n                    Nothing ->\n                        onEffects router rest state\n                \n\ntype SelfMsg\n    = Cleanup StreamRequest\n\n\nonSelfMsg : Platform.Router msg SelfMsg -> SelfMsg -> RequestsState msg -> Task Never (RequestsState msg)\nonSelfMsg _ event state =\n    case event of\n        Cleanup (StreamRequest reqId) ->\n            Task.succeed \n                { state | msgHandlers = Dict.remove reqId state.msgHandlers }\n",
        "effect module Terminal where { subscription = TerminalSub } exposing\n    ( Permission\n    , Configuration\n    , Size\n    , initialize\n    --\n    , setStdInRawMode\n    , setProcessTitle\n    --\n    , onResize\n    )\n\n\n{-| This lets you interact with the user's terminal, if an interactive\nterminal is connected to this application.\n\n## Initialization\n\n@docs Permission, Configuration, Size, initialize\n\n## Commands\n\n@docs setStdInRawMode, setProcessTitle\n\n## Subscriptions\n\n@docs onResize\n\n-}\n\n\nimport Task exposing (Task)\nimport Process\nimport Gren.Kernel.Terminal\nimport Init\nimport Internal.Init \n\n\n{-| The permission for performing commands specified in this module.\n-}\ntype Permission\n    = Permission\n\n\n{-| The configuration of the attached interactive terminal.\n-}\ntype alias Configuration =\n    { permission : Permission\n    , colorDepth : Int\n    , columns : Int\n    , rows : Int\n    }\n\n\n{-| Size of a terminal. Handy to know for drawing a text-based UI.\n-}\ntype alias Size =\n    { columns : Int\n    , rows : Int\n    }\n\n\n-- INIT\n\n\n{-| Initializes the `Terminal` subsystem.\n\n`Nothing` is returned if this program isn't connected to an interactive terminal, which\ncan happen in CI-setups or when used as part of a unix pipe.\n-}\ninitialize : Init.Task (Maybe Configuration)\ninitialize =\n    Gren.Kernel.Terminal.init\n        |> Task.map (\\raw ->\n            if raw.isTTY then\n                Just\n                    { permission = Permission\n                    , colorDepth = raw.colorDepth\n                    , columns = raw.columns\n                    , rows = raw.rows\n                    }\n\n            else\n                Nothing\n        )\n        |> Internal.Init.Task\n\n\n-- COMMANDS\n\n\n{-| In it's default mode, `stdin` only sends data when the user hits the enter key.\n\nIf you switch over to raw mode, every keypress will be sent over the stream, and special\ncombinations like `Ctrl-C` will no longer trigger the kill signal.\n\nEnable this when you need full control over how input is handled.\n-}\nsetStdInRawMode : Permission -> Bool -> Task x {}\nsetStdInRawMode _ toggle =\n    Gren.Kernel.Terminal.setStdInRawMode toggle\n\n\n{-| Set the title of the running process. This will usually display in\nactivity monitors or in the title bar of your terminal emulator.\n-}\nsetProcessTitle : Permission -> String -> Task x {}\nsetProcessTitle _ title =\n    Gren.Kernel.Terminal.setProcessTitle title\n\n\n-- SUBSCRIPTIONS\n\n\ntype TerminalSub msg\n    = OnResize (Size -> msg)\n\n\nsubMap : (a -> b) -> TerminalSub a -> TerminalSub b\nsubMap mapFn sub =\n    case sub of\n        OnResize msgMap ->\n            OnResize (mapFn << msgMap)\n\n\n{-| A subscription that triggers every time the size of the terminal changes.\n-}\nonResize : Permission -> (Size -> msg) -> Sub msg\nonResize _ toMsg =\n    subscription (OnResize toMsg)\n\n\n-- LOOP\n\n\ntype alias State msg =\n    { taggers : Array (Size -> msg)\n    , maybeProcessId : Maybe Process.Id\n    }\n\n\ninit : Task Never (State msg)\ninit =\n    Task.succeed\n        { taggers = []\n        , maybeProcessId = Nothing\n        }\n\n\nonEffects\n    : Platform.Router msg Event\n    -> Array (TerminalSub msg)\n    -> State msg\n    -> Task Never (State msg)\nonEffects router subs state =\n    let\n        newTaggers =\n            Array.foldl extractTagger [] subs\n\n        extractTagger (OnResize tagger) acc =\n            Array.pushLast tagger acc\n\n        subscriptionTask =\n            if Array.length newTaggers > 0 then\n                case state.maybeProcessId of\n                    Just pid ->\n                        Task.succeed <| Just pid\n\n                    Nothing ->\n                        Gren.Kernel.Terminal.attachListener (\\data -> Platform.sendToSelf router (SelfOnResize data))\n                            |> Process.spawn\n                            |> Task.map Just\n\n\n            else\n                case state.maybeProcessId of\n                    Just pid ->\n                        Process.kill pid\n                            |> Task.map (\\_ -> Nothing)\n\n                    Nothing ->\n                        Task.succeed Nothing\n    in\n    subscriptionTask\n        |> Task.andThen (\\maybeProcessId ->\n            Task.succeed\n                { taggers = newTaggers\n                , maybeProcessId = maybeProcessId\n                }\n            )\n\n\ntype Event\n    = SelfOnResize Size\n\n\nonSelfMsg : Platform.Router msg Event -> Event -> State msg -> Task Never (State msg)\nonSelfMsg router event state =\n    case event of\n        SelfOnResize newSize ->\n            state.taggers\n                |> Array.map (\\tagger -> tagger newSize)\n                |> Array.foldl\n                    (\\msg tasks ->\n                        Task.andThen (\\{} -> Platform.sendToApp router msg) tasks\n                    )\n                    (Task.succeed {})\n                |> Task.map (\\_ -> state)\n",
        "module Maybe exposing\n    ( Maybe(..)\n    , withDefault, map, map2, map3, map4, map5\n    , andThen\n    )\n\n{-| This library fills a bunch of important niches in Gren. A `Maybe` can help\nyou with optional arguments, error handling, and records with optional fields.\n\n\n@docs Maybe\n\n\n## Common Helpers\n\n@docs withDefault, map, map2, map3, map4, map5\n\n\n## Chaining Maybes\n\n@docs andThen\n\n-}\n\nimport Basics exposing (Bool(..))\n\n\n{-| Represent values that may or may not exist. It can be useful if you have a\nrecord field that is only filled in sometimes. Or if a function takes a value\nsometimes, but does not absolutely need it.\n\n    -- A person, but maybe we do not know their age.\n    type alias Person =\n        { name : String\n        , age : Maybe Int\n        }\n\n    tom =\n        { name = \"Tom\", age = Just 42 }\n\n    sue =\n        { name = \"Sue\", age = Nothing }\n\n-}\ntype Maybe a\n    = Just a\n    | Nothing\n\n\n{-| Provide a default value, turning an optional value into a normal\nvalue. This comes in handy when paired with functions like\n[`Dict.get`](Dict#get) which gives back a `Maybe`.\n\n    withDefault 100 (Just 42) -- 42\n\n    withDefault 100 Nothing -- 100\n\n    withDefault \"unknown\" (Dict.get \"Tom\" Dict.empty) -- \"unknown\"\n\n**Note:** This can be overused! Many cases are better handled by a `case`\nexpression. And if you end up using `withDefault` a lot, it can be a good sign\nthat a [custom type][ct] will clean your code up quite a bit!\n\n[ct]: https://guide.gren-lang.org/types/custom_types.html\n\n-}\nwithDefault : a -> Maybe a -> a\nwithDefault default maybe =\n    case maybe of\n        Just value ->\n            value\n\n        Nothing ->\n            default\n\n\n{-| Transform a `Maybe` value with a given function:\n\n    map sqrt (Just 9) == Just 3\n\n    map sqrt Nothing == Nothing\n\n    map sqrt (String.toFloat \"9\") == Just 3\n\n    map sqrt (String.toFloat \"x\") == Nothing\n\n-}\nmap : (a -> b) -> Maybe a -> Maybe b\nmap f maybe =\n    case maybe of\n        Just value ->\n            Just (f value)\n\n        Nothing ->\n            Nothing\n\n\n{-| Apply a function if all the arguments are `Just` a value.\n\n    map2 (+) (Just 3) (Just 4) == Just 7\n\n    map2 (+) (Just 3) Nothing == Nothing\n\n    map2 (+) Nothing (Just 4) == Nothing\n\n    map2 (+) (String.toInt \"1\") (String.toInt \"123\") == Just 124\n\n    map2 (+) (String.toInt \"x\") (String.toInt \"123\") == Nothing\n\n    map2 (+) (String.toInt \"1\") (String.toInt \"1.3\") == Nothing\n\n-}\nmap2 : (a -> b -> value) -> Maybe a -> Maybe b -> Maybe value\nmap2 func ma mb =\n    case ma of\n        Nothing ->\n            Nothing\n\n        Just a ->\n            case mb of\n                Nothing ->\n                    Nothing\n\n                Just b ->\n                    Just (func a b)\n\n\n{-| -}\nmap3 : (a -> b -> c -> value) -> Maybe a -> Maybe b -> Maybe c -> Maybe value\nmap3 func ma mb mc =\n    case ma of\n        Nothing ->\n            Nothing\n\n        Just a ->\n            case mb of\n                Nothing ->\n                    Nothing\n\n                Just b ->\n                    case mc of\n                        Nothing ->\n                            Nothing\n\n                        Just c ->\n                            Just (func a b c)\n\n\n{-| -}\nmap4 : (a -> b -> c -> d -> value) -> Maybe a -> Maybe b -> Maybe c -> Maybe d -> Maybe value\nmap4 func ma mb mc md =\n    case ma of\n        Nothing ->\n            Nothing\n\n        Just a ->\n            case mb of\n                Nothing ->\n                    Nothing\n\n                Just b ->\n                    case mc of\n                        Nothing ->\n                            Nothing\n\n                        Just c ->\n                            case md of\n                                Nothing ->\n                                    Nothing\n\n                                Just d ->\n                                    Just (func a b c d)\n\n\n{-| -}\nmap5 : (a -> b -> c -> d -> e -> value) -> Maybe a -> Maybe b -> Maybe c -> Maybe d -> Maybe e -> Maybe value\nmap5 func ma mb mc md me =\n    case ma of\n        Nothing ->\n            Nothing\n\n        Just a ->\n            case mb of\n                Nothing ->\n                    Nothing\n\n                Just b ->\n                    case mc of\n                        Nothing ->\n                            Nothing\n\n                        Just c ->\n                            case md of\n                                Nothing ->\n                                    Nothing\n\n                                Just d ->\n                                    case me of\n                                        Nothing ->\n                                            Nothing\n\n                                        Just e ->\n                                            Just (func a b c d e)\n\n\n{-| Chain together many computations that may fail. It is helpful to see its\ndefinition:\n\n    andThen : (a -> Maybe b) -> Maybe a -> Maybe b\n    andThen callback maybe =\n        case maybe of\n            Just value ->\n                callback value\n\n            Nothing ->\n                Nothing\n\nThis means we only continue with the callback if things are going well. For\nexample, say you need to parse some user input as a month:\n\n    parseMonth : String -> Maybe Int\n    parseMonth userInput =\n        String.toInt userInput\n            |> andThen toValidMonth\n\n    toValidMonth : Int -> Maybe Int\n    toValidMonth month =\n        if 1 <= month && month <= 12 then\n            Just month\n\n        else\n            Nothing\n\nIn the `parseMonth` function, if `String.toInt` produces `Nothing` (because\nthe `userInput` was not an integer) this entire chain of operations will\nshort-circuit and result in `Nothing`. If `toValidMonth` results in `Nothing`,\nagain the chain of computations will result in `Nothing`.\n\n-}\nandThen : (a -> Maybe b) -> Maybe a -> Maybe b\nandThen callback maybeValue =\n    case maybeValue of\n        Just value ->\n            callback value\n\n        Nothing ->\n            Nothing\n\n\n\n-- FOR INTERNAL USE ONLY\n--\n-- Use `case` expressions for this in Gren code!\n\n\nisJust : Maybe a -> Bool\nisJust maybe =\n    case maybe of\n        Just _ ->\n            True\n\n        Nothing ->\n            False\n\n\ndestruct : b -> (a -> b) -> Maybe a -> b\ndestruct default func maybe =\n    case maybe of\n        Just a ->\n            func a\n\n        Nothing ->\n            default\n",
        "module Bytes.Encode exposing\n  ( encode\n  , Encoder\n  , signedInt8, signedInt16, signedInt32\n  , unsignedInt8, unsignedInt16, unsignedInt32\n  , float32, float64\n  , bytes\n  , sequence\n  )\n\n\n{-| Functions for turning things into bytes.\n\n@docs Encoder, encode, sequence\n\n## Integers\n@docs signedInt8, signedInt16, signedInt32, unsignedInt8, unsignedInt16, unsignedInt32\n\n## Floats\n@docs float32, float64\n\n## Bytes\n@docs bytes\n\n-}\n\n\nimport Array exposing (Array)\nimport Basics exposing (..)\nimport Bytes exposing (Bytes, Endianness(..))\nimport Maybe exposing (Maybe(..))\nimport String exposing (String)\n\n\n\n-- ENCODER\n\n\n{-| Describes how to generate a sequence of bytes.\n\nThese encoders snap together with [`sequence`](#sequence) so you can start with\nsmall building blocks and put them together into a more complex encoding.\n-}\ntype Encoder\n  = I8 Int\n  | I16 Endianness Int\n  | I32 Endianness Int\n  | U8 Int\n  | U16 Endianness Int\n  | U32 Endianness Int\n  | F32 Endianness Float\n  | F64 Endianness Float\n  | Seq Int (Array Encoder)\n  | Utf8 Int String\n  | Bytes Bytes\n\n\n\n-- ENCODE\n\n\n{-| Turn an `Encoder` into `Bytes`.\n\n    encode (unsignedInt8     7) -- <07>\n    encode (unsignedInt16 BE 7) -- <0007>\n    encode (unsignedInt16 LE 7) -- <0700>\n\nThe `encode` function is designed to minimize allocation. It figures out the\nexact length necessary to fit everything in `Bytes` and then generate that\nvalue directly. This is valuable when you are encoding more elaborate data:\n\n    import Bytes exposing (Endianness(..))\n    import Bytes.Encode as Encode\n\n    type alias Person =\n      { age : Int\n      , name : String\n      }\n\n    toEncoder : Person -> Encode.Encoder\n    toEncoder person =\n      Encode.sequence\n        [ Encode.unsignedInt16 BE person.age\n        , Encode.unsignedInt16 BE (Encode.getStringWidth person.name)\n        , Encode.string person.name\n        ]\n\n    -- encode (toEncoder ({ age = 33, name = \"Tom\" })) == <00210003546F6D>\n\nDid you know it was going to be seven bytes? How about when you have a hundred\npeople to serialize? And when some have Japanese and Norwegian names? Having\nthis intermediate `Encoder` can help reduce allocation quite a lot!\n-}\nencode : Encoder -> Bytes\nencode =\n  Gren.Kernel.Bytes.encode\n\n\n\n-- INTEGERS\n\n\n{-| Encode integers from `-128` to `127` in one byte.\n-}\nsignedInt8 : Int -> Encoder\nsignedInt8 =\n  I8\n\n\n{-| Encode integers from `-32768` to `32767` in two bytes.\n-}\nsignedInt16 : Endianness -> Int -> Encoder\nsignedInt16 =\n  I16\n\n\n{-| Encode integers from `-2147483648` to `2147483647` in four bytes.\n-}\nsignedInt32 : Endianness -> Int -> Encoder\nsignedInt32 =\n  I32\n\n\n{-| Encode integers from `0` to `255` in one byte.\n-}\nunsignedInt8 : Int -> Encoder\nunsignedInt8 =\n  U8\n\n\n{-| Encode integers from `0` to `65535` in two bytes.\n-}\nunsignedInt16 : Endianness -> Int -> Encoder\nunsignedInt16 =\n  U16\n\n\n{-| Encode integers from `0` to `4294967295` in four bytes.\n-}\nunsignedInt32 : Endianness -> Int -> Encoder\nunsignedInt32 =\n  U32\n\n\n\n-- FLOATS\n\n\n{-| Encode 32-bit floating point numbers in four bytes.\n-}\nfloat32 : Endianness -> Float -> Encoder\nfloat32 =\n  F32\n\n\n{-| Encode 64-bit floating point numbers in eight bytes.\n-}\nfloat64 : Endianness -> Float -> Encoder\nfloat64 =\n  F64\n\n\n\n-- BYTES\n\n\n{-| Copy bytes directly into the new `Bytes` sequence. This does not record the\nlength though! You usually want to say something like this:\n\n    import Bytes exposing (Bytes, Endianness(..))\n    import Bytes.Encode as Encode\n\n    png : Bytes -> Encode.Encoder\n    png imageData =\n      Encode.sequence\n        [ Encode.unsignedInt32 BE (Bytes.length imageData)\n        , Encode.bytes imageData\n        ]\n\nThis allows you to represent the length however is necessary for your protocol.\nFor example, you can use [Base 128 Varints][pb] for ProtoBuf,\n[Variable-Length Integers][sql] for SQLite, or whatever else they dream up.\n\n[pb]: https://developers.google.com/protocol-buffers/docs/encoding#varints\n[sql]: https://www.sqlite.org/src4/doc/trunk/www/varint.wiki\n-}\nbytes : Bytes -> Encoder\nbytes =\n  Bytes\n\n\n-- SEQUENCE\n\n\n{-| Put together a bunch of builders. So if you wanted to encode three `Float`\nvalues for the position of a ball in 3D space, you could say:\n\n    import Bytes exposing (Endianness(..))\n    import Bytes.Encode as Encode\n\n    type alias Ball = { x : Float, y : Float, z : Float }\n\n    ball : Ball -> Encode.Encoder\n    ball {x,y,z} =\n      Encode.sequence\n        [ Encode.float32 BE x\n        , Encode.float32 BE y\n        , Encode.float32 BE z\n        ]\n\n-}\nsequence : Array Encoder -> Encoder\nsequence builders =\n  Seq (getLengths 0 builders) builders\n\n\n-- WRITE\n\n\nwrite : Encoder -> Bytes -> Int -> Int\nwrite builder mb offset =\n  case builder of\n    I8    n -> Gren.Kernel.Bytes.write_i8  mb offset n\n    I16 e n -> Gren.Kernel.Bytes.write_i16 mb offset n (e == LE)\n    I32 e n -> Gren.Kernel.Bytes.write_i32 mb offset n (e == LE)\n    U8    n -> Gren.Kernel.Bytes.write_u8  mb offset n\n    U16 e n -> Gren.Kernel.Bytes.write_u16 mb offset n (e == LE)\n    U32 e n -> Gren.Kernel.Bytes.write_u32 mb offset n (e == LE)\n    F32 e n -> Gren.Kernel.Bytes.write_f32 mb offset n (e == LE)\n    F64 e n -> Gren.Kernel.Bytes.write_f64 mb offset n (e == LE)\n    Seq _ bs -> writeSequence bs mb offset\n    Utf8 _ s -> Gren.Kernel.Bytes.write_string mb offset s\n    Bytes bs -> Gren.Kernel.Bytes.write_bytes mb offset bs\n\n\nwriteSequence : Array Encoder -> Bytes -> Int -> Int\nwriteSequence builders mb offset =\n  Array.foldl\n    (\\builder currentOffset ->\n      write builder mb currentOffset\n    )\n    offset\n    builders\n\n\n-- LENGTHS\n\n\ngetLength : Encoder -> Int\ngetLength builder =\n  case builder of\n    I8    _ -> 1\n    I16 _ _ -> 2\n    I32 _ _ -> 4\n    U8    _ -> 1\n    U16 _ _ -> 2\n    U32 _ _ -> 4\n    F32 _ _ -> 4\n    F64 _ _ -> 8\n    Seq w _ -> w\n    Utf8 w _ -> w\n    Bytes bs -> Gren.Kernel.Bytes.length bs\n\n\ngetLengths : Int -> Array Encoder -> Int\ngetLengths length builders =\n  Array.foldl \n    (\\builder sum ->\n      sum + getLength builder\n    )\n    length\n    builders\n",
        "module Bytes exposing\n  ( Bytes\n  , empty\n  , isEmpty\n  , length\n  --\n  , Endianness(..)\n  , getHostEndianness\n  --\n  , fromString\n  , toString\n  --\n  , join\n  )\n\n\n{-| Functions for working with sequences of bytes.\n\n@docs Bytes, empty, isEmpty, length\n\n## Endianness\n@docs Endianness, getHostEndianness\n\n## Strings\n@docs fromString, toString\n\n## Combine\n@docs join\n\n-}\n\n\nimport Array exposing (Array)\nimport Basics exposing (..)\nimport Maybe exposing (Maybe)\nimport String exposing (String)\nimport Task exposing (Task)\nimport Gren.Kernel.Bytes\n\n\n-- BYTES\n\n\n{-| A sequence of bytes.\n\nA byte is a chunk of eight bits. For example, the letter `j` is usually\nrepresented as the byte `01101010`, and the letter `k` is `01101011`.\n\nSeeing each byte as a stream of zeros and ones can be quite confusing though,\nso it is common to use hexidecimal numbers instead:\n\n```\n| Binary | Hex |\n+--------+-----+\n|  0000  |  0  |\n|  0001  |  1  |\n|  0010  |  2  |\n|  0011  |  3  |     j = 01101010\n|  0100  |  4  |         \\__/\\__/\n|  0101  |  5  |           |   |\n|  0110  |  6  |           6   A\n|  0111  |  7  |\n|  1000  |  8  |     k = 01101011\n|  1001  |  9  |         \\__/\\__/\n|  1010  |  A  |           |   |\n|  1011  |  B  |           6   B\n|  1100  |  C  |\n|  1101  |  D  |\n|  1110  |  E  |\n|  1111  |  F  |\n```\n\nSo `j` is `6A` and `k` is `6B` in hexidecimal. This more compact representation\nis great when you have a sequence of bytes. You can see this even in a short\nstring like `\"jazz\"`:\n\n```\nbinary                                 hexidecimal\n01101010 01100001 01111010 01111010 => 6A 61 7A 7A\n```\n\nAnyway, the point is that `Bytes` is a sequence of bytes!\n-}\ntype Bytes = Bytes\n\n\n{-| The sequence with exactly 0 bytes.\n-}\nempty : Bytes\nempty =\n  Gren.Kernel.Bytes.empty\n\n\n{-| Check if the given byte sequence has 0 bytes in it.\n-}\nisEmpty : Bytes -> Bool\nisEmpty bytes =\n  length bytes == 0\n\n\n{-| Get the length of a sequence of bytes.\n\nSo if a sequence has four-hundred bytes, then `length bytes` would give back\n`400`. That may be 400 unsigned 8-bit integers, 100 signed 32-bit integers, or\neven a UTF-8 string. The content does not matter. This is just figuring out\nhow many bytes there are!\n-}\nlength : Bytes -> Int\nlength =\n  Gren.Kernel.Bytes.length\n\n\n\n-- ENDIANNESS\n\n\n{-| Different computers store integers and floats slightly differently in\nmemory. Say we have the integer `0x1A2B3C4D` in our program. It needs four\nbytes (32 bits) in memory. It may seem reasonable to lay them out in order:\n\n```\n   Big-Endian (BE)      (Obvious Order)\n+----+----+----+----+\n| 1A | 2B | 3C | 4D |\n+----+----+----+----+\n```\n\nBut some people thought it would be better to store the bytes in the opposite\norder:\n\n```\n  Little-Endian (LE)    (Shuffled Order)\n+----+----+----+----+\n| 4D | 3C | 2B | 1A |\n+----+----+----+----+\n```\n\nNotice that **the _bytes_ are shuffled, not the bits.** It is like if you cut a\nphoto into four strips and shuffled the strips. It is not a mirror image.\nThe theory seems to be that an 8-bit `0x1A` and a 32-bit `0x0000001A` both have\n`1A` as the first byte in this scheme. Maybe this was helpful when processors\nhandled one byte at a time.\n\n**Most processors use little-endian (LE) layout.** This seems to be because\nIntel did it this way, and other chip manufactures followed their convention.\n**Most network protocols use big-endian (BE) layout.** I suspect this is\nbecause if you are trying to debug a network protocol, it is nice if your\nintegers are not all shuffled.\n\n**Note:** Endianness is relevant for integers and floats, but not strings.\nUTF-8 specifies the order of bytes explicitly.\n\n**Note:** The terms little-endian and big-endian are a reference to an egg joke\nin Gulliver's Travels. They first appeared in 1980 in [this essay][essay], and\nyou can decide for yourself if they stood the test of time. I personally find\nthese terms quite unhelpful, so I say “Obvious Order” and “Shuffled Order” in\nmy head. I remember which is more common by asking myself, “if things were\nobvious, would I have to ask this question?”\n\n[essay]: http://www.ietf.org/rfc/ien/ien137.txt\n-}\ntype Endianness = LE | BE\n\n\n{-| Is this program running on a big-endian or little-endian machine?\n-}\ngetHostEndianness : Task x Endianness\ngetHostEndianness =\n  Gren.Kernel.Bytes.getHostEndianness LE BE\n\n\n-- STRINGS\n\n\n{-| Convert a `String` to a `Bytes`. The resulting bytes will be in the\nUTF-8 encoding.\n\nSome characters take one byte, while others can take up to four. Read more\nabout [UTF-8](https://en.wikipedia.org/wiki/UTF-8) to learn the details!\n-}\nfromString : String -> Bytes\nfromString =\n  Gren.Kernel.Bytes.fromString\n\n\n{-| Convert UTF-8 encoded `Bytes` to `String`. If the byte sequence isn't\nvalid UTF-8, `Nothing` will be returned.\n-}\ntoString : Bytes -> Maybe String\ntoString =\n  Gren.Kernel.Bytes.toString\n\n\n-- COMBINE\n\n\n{-| Join all `Bytes` in an `Array` into a single `Bytes`.\n-}\njoin : Array Bytes -> Bytes\njoin =\n  Gren.Kernel.Bytes.join\n",
        "effect module Stream where { subscription = StreamSub } exposing\n    ( Stream\n    , send\n    , sendString\n    , sendLine\n    , listen\n    )\n\n{-| A `Stream` is an abstract flow of data from one point to another,\nlike bytes flowing from a server to a client, or from a file to a program.\n\n@docs Stream, send, sendString, sendLine, listen\n\n-}\n\nimport Dict exposing (Dict)\nimport Gren.Kernel.Stream\nimport Task exposing (Task)\nimport Bytes exposing (Bytes)\nimport Process\nimport Internal.Stream as Internal\n\n\n{-| A specific stream.\n-}\ntype alias Stream\n    = Internal.Stream\n\n\n{-| This lets you send bytes over a `Stream`.\n-}\nsend : Stream -> Bytes -> Task x {}\nsend (Internal.Stream _ kernelStream) bytes =\n    Gren.Kernel.Stream.send kernelStream bytes\n\n\n{-| This lets you send a string over a `Stream`.\n-}\nsendString : Stream -> String -> Task x {}\nsendString stream string =\n    send stream <| Bytes.fromString string\n\n\n{-| This sends string and a newline over a `Stream`.\n-}\nsendLine : Stream -> String -> Task x {}\nsendLine stream string =\n    sendString stream <| string ++ \"\\n\"\n\n\n-- SUBSCRIPTION\n\n\ntype StreamSub msg\n    = Listen Stream (Bytes -> msg)\n\n\nsubMap : (a -> b) -> StreamSub a -> StreamSub b\nsubMap mapFn sub =\n    case sub of\n        Listen stream msgMap ->\n            Listen stream (mapFn << msgMap)\n\n\n{-| This notifies your application every time bytes have arrived over the `Stream`.\n-}\nlisten : Stream -> (Bytes -> msg) -> Sub msg\nlisten stream msgMap =\n    subscription (Listen stream msgMap)\n\n\n-- LOOP\n\n\ntype alias State msg =\n    { taggers : Dict Int (Tagger msg)\n    , processes : Dict Int Process.Id\n    }\n\n\ntype alias Tagger msg =\n    { stream : Stream\n    , taggers : Array (Bytes -> msg)\n    }\n\n\ninit : Task.Task Never (State msg)\ninit =\n    Task.succeed\n        { taggers = Dict.empty\n        , processes = Dict.empty\n        }\n\n\nonEffects\n    : Platform.Router msg Event\n    -> Array (StreamSub msg)\n    -> State msg\n    -> Task.Task Never (State msg)\nonEffects router subs state =\n    let\n        newTaggers =\n            Array.foldl subToListener Dict.empty subs\n\n        toSpawn =\n            Dict.diff newTaggers state.taggers\n                |> Dict.values\n                |> Array.map .stream\n\n        existingProcesses =\n            Dict.filter (\\sid _ -> Dict.member sid newTaggers) state.processes\n\n        killTasks =\n            Dict.diff state.processes newTaggers\n                |> Dict.values\n                |> Array.foldl\n                    (\\id tasks -> Task.andThen (\\{} -> Process.kill id) tasks)\n                    (Task.succeed {})\n    in\n    killTasks\n        |> Task.andThen (\\_ -> spawnHelp router toSpawn existingProcesses)\n        |> Task.map (\\newProcesses ->\n            { taggers = newTaggers\n            , processes = newProcesses\n            }\n        )\n\n\nsubToListener : StreamSub msg -> Dict Int (Tagger msg) -> Dict Int (Tagger msg)\nsubToListener sub taggers =\n    case sub of\n        Listen ((Internal.Stream sid _) as stream) tagger ->\n            case Dict.get sid taggers of\n                Just data ->\n                    Dict.set sid\n                        { data | taggers = Array.pushLast tagger data.taggers }\n                        taggers\n\n                Nothing ->\n                    Dict.set sid\n                        { stream = stream\n                        , taggers = [ tagger ]\n                        }\n                        taggers\n\n\nspawnHelp : Platform.Router msg Event -> Array Stream -> Dict Int Process.Id -> Task.Task Never (Dict Int Process.Id)\nspawnHelp router toSpawn processes =\n  case Array.first toSpawn of\n    Nothing ->\n      Task.succeed processes\n\n    Just ((Internal.Stream sid rawStream) as nextStream) ->\n      let\n        spawnStream =\n          Process.spawn <|\n            attachListener rawStream (\\data ->\n                Platform.sendToSelf router (FromStream nextStream data)\n            )\n\n        rest =\n          Array.dropFirst 1 toSpawn\n\n        spawnRest processId =\n          spawnHelp router rest (Dict.set sid processId processes)\n      in\n        spawnStream\n          |> Task.andThen spawnRest\n\n\nattachListener : Internal.RawStream -> (Bytes -> Task.Task Never {}) -> Task.Task x Never\nattachListener =\n  Gren.Kernel.Stream.attachListener\n\n\ntype Event\n    = FromStream Stream Bytes\n\n\nonSelfMsg : Platform.Router msg Event -> Event -> State msg -> Task.Task Never (State msg)\nonSelfMsg router event state =\n    case event of\n        FromStream (Internal.Stream sid _) data ->\n            Dict.get sid state.taggers\n                |> Maybe.map .taggers\n                |> Maybe.withDefault []\n                |> Array.map (\\tagger -> tagger data)\n                |> Array.foldl\n                    (\\msg tasks ->\n                        Task.andThen (\\{} -> Platform.sendToApp router msg) tasks\n                    )\n                    (Task.succeed {})\n                |> Task.map (\\_ -> state)\n",
        "effect module HttpServer where { subscription = HttpSub } exposing\n    -- Init\n    ( Permission\n    , initialize\n\n    -- Server\n    , Server\n    , ServerError(..)\n    , createServer\n\n    -- Requests\n    , Request\n    , Method(..)\n    , methodToString\n    , bodyAsString\n    , bodyFromJson\n    , requestInfo\n    , onRequest\n    )\n\n{-| Create a server that can respond to HTTP requests.\n\nYou write your server using The Elm Architecture by subscribing to request\nevents and responding with commands in update.\n\nSee [examples/http-server](https://github.com/gren-lang/node/tree/main/examples/http-server) for a working example.\n\n## Initialization\n\n@docs Permission, Server, ServerError, initialize, createServer\n\n## Requests\n\n@docs Request, Method, methodToString, bodyAsString, bodyFromJson, requestInfo\n\n## Responding to requests\n\n@docs onRequest\n\nSee [HttpServer.Response](HttpServer.Response) for more details on responding to requests.\n\n-}\n\nimport Bytes exposing (Bytes)\nimport Bytes.Decode as Decode\nimport Dict exposing (Dict)\nimport Init\nimport Internal.Init\nimport Json.Encode\nimport Json.Decode\nimport Node\nimport Task exposing (Task)\nimport Gren.Kernel.HttpServer\nimport HttpServer.Response exposing (Response(..))\nimport Url exposing (Url, Protocol(..))\n\n\n-- INITIALIZATION\n\n\n{-| The permission to start a [`Server`](HttpServer.Server).\n\nYou get this from [`initialize`](HttpServer.initialize).\n-}\ntype Permission\n    = Permission\n\n{-| The HTTP server.\n-}\ntype Server\n    -- Note: Actual implementation in Kernel code\n    = Server\n\n\n{-| Error code and message from node.\nMost likely from a failed attempt to start the server (e.g. `EADDRINUSE`).\nRefer to the [node docs](https://nodejs.org/docs/latest-v18.x/api/errors.html) for details.\n-}\ntype ServerError =\n    ServerError String String\n\n\n{-| Initialize the [`HttpServer`](HttpServer) module and get permission to create a server.\n-}\ninitialize : Init.Task Permission\ninitialize =\n    Task.succeed Permission\n        |> Internal.Init.Task\n\n\n\n{-| Task to initialize a [`Server`](HttpServer#Server).\n-}\ncreateServer : Permission -> { host : String, port_ : Int } -> Task ServerError Server\ncreateServer _ options =\n    Gren.Kernel.HttpServer.createServer options.host options.port_\n\n\n-- REQUESTS\n\n\n{-| An incoming HTTP reqest.\n-}\ntype alias Request =\n    { headers : Dict String String\n    , method : Method\n    , body : Bytes\n    , url : Url\n   }\n\n\n{-| HTTP request methods.\n-}\ntype Method\n    = GET\n    | HEAD\n    | POST\n    | PUT\n    | DELETE\n    | CONNECT\n    | TRACE\n    | PATCH\n    | UNKNOWN String\n\n\n{-| String representation of method\n-}\nmethodToString : Method -> String\nmethodToString method =\n    case method of\n        GET ->\n            \"GET\"\n        \n        HEAD ->\n            \"HEAD\"\n        \n        POST ->\n            \"POST\"\n        \n        PUT ->\n            \"PUT\"\n        \n        DELETE ->\n            \"DELETE\"\n        \n        CONNECT ->\n            \"CONNECT\"\n        \n        TRACE ->\n            \"TRACE\"\n        \n        PATCH ->\n            \"PATCH\"\n        \n        UNKNOWN value ->\n            value\n\n\n{-| Turn the pieces of a request into a [`Request`](HttpServer.Request) record.\n\nThis is only used internally.\n-}\ntoRequest :\n    { urlProtocol : String\n    , urlHost : String\n    , urlPort : String\n    , urlPath : String\n    , urlQuery : String\n    , urlFragment : String\n    , headers : Array String\n    , method : String\n    , body : Bytes\n    }\n    -> Request\ntoRequest \n    { urlProtocol\n    , urlHost\n    , urlPort\n    , urlPath\n    , urlQuery\n    , urlFragment\n    , headers\n    , method\n    , body \n    } =\n    { method = toMethod method\n    , body = body\n    , headers =\n        headers\n            |> arrayPairs\n            |> dictFromPairs\n    , url =\n        { protocol = \n            if urlProtocol == \"https:\" then\n                Https\n            else\n                Http\n        , host = urlHost\n        , path = urlPath\n        , port_ = String.toInt urlPort\n        , query = \n            if urlQuery == \"\" then\n                Nothing\n            else\n                Just urlQuery\n        , fragment =\n            if urlFragment == \"\" then\n                Nothing\n            else\n                Just urlFragment\n        }\n    }\n\n\n{-| Get request body as a string.\n-}\nbodyAsString : Request -> Maybe String\nbodyAsString req =\n    Bytes.toString req.body\n\n\n{-| Get request body as json.\n-}\nbodyFromJson : Json.Decode.Decoder a -> Request -> Result Json.Decode.Error a\nbodyFromJson decoder req =\n    req\n        |> bodyAsString\n        |> Maybe.withDefault \"\" -- or better if result holds a maybe?\n        |> Json.Decode.decodeString decoder\n\n\n{-| Get a string representation of the request.\n\nGood for logging.\n-}\nrequestInfo : Request -> String\nrequestInfo req =\n    let\n        method\n            = case req.method of\n                GET ->\n                    \"GET\"\n\n                HEAD ->\n                    \"HEAD\"\n\n                POST ->\n                    \"POST\"\n\n                PUT ->\n                    \"PUT\"\n\n                DELETE ->\n                    \"DELETE\"\n\n                CONNECT ->\n                    \"CONNECT\"\n\n                TRACE ->\n                    \"TRACE\"\n\n                PATCH ->\n                    \"PATCH\"\n\n                UNKNOWN m ->\n                    \"UNKNOWN(\" ++ m ++ \")\"\n    in\n    method ++ \" \" ++ req.url.path\n\n\ntoMethod : String -> Method\ntoMethod s =\n    case s of\n        \"GET\" ->\n            GET\n        \n        \"HEAD\" ->\n            HEAD\n\n        \"POST\" ->\n            POST\n        \n        \"PUT\" ->\n            PUT\n        \n        \"DELETE\" ->\n            DELETE\n        \n        \"CONNECT\" ->\n            CONNECT\n        \n        \"TRACE\" ->\n            TRACE\n        \n        \"PATCH\" ->\n            PATCH\n        \n        _ ->\n            UNKNOWN s\n\n\narrayPairs : Array String -> Array (Array String)\narrayPairs a =\n    let\n        pair = \n            Array.takeFirst 2 a\n        \n        rest =\n            Array.dropFirst 2 a\n        \n        allPairs =\n            [ pair ] ++ case rest of\n                [] ->\n                    []\n                \n                _ ->\n                    arrayPairs rest\n    in\n    allPairs\n\n\ndictFromPairs : Array (Array String) -> Dict String String\ndictFromPairs pairs =\n    let\n        mapper p dict =\n            case p of\n                [a, b] ->\n                    Dict.set a b dict\n                \n                _ ->\n                    dict\n    in\n    Array.foldl mapper Dict.empty pairs\n\n\n-- EFFECT STUFF\n\n\ntype HttpSub msg\n    = OnRequestSub Server (Request -> Response -> msg)\n\n\nsubMap : (a -> b) -> HttpSub a -> HttpSub b\nsubMap f sub =\n    case sub of\n        OnRequestSub server msg ->\n            OnRequestSub server (\\req res -> f (msg req res))\n\n\ntype alias State msg =\n    Array (HttpSub msg)\n\n\ninit : Task Never (State msg)\ninit =\n    Task.succeed []\n\n\nonEffects\n    : Platform.Router msg SelfMsg\n    -> Array (HttpSub msg)\n    -> State msg\n    -> Task Never (State msg)\nonEffects router subs state =\n    let\n        _removeListeners =\n            state\n                |> Array.map\n                    (\\(OnRequestSub server _) ->\n                        Gren.Kernel.HttpServer.removeAllListeners server\n                    )\n        \n        _addListeners =\n            subs\n                |> Array.map\n                    (\\(OnRequestSub server msg) ->\n                        Gren.Kernel.HttpServer.addListener server router msg\n                    )\n    in\n    Task.succeed subs\n\n\ntype SelfMsg =\n    Never\n\n\nonSelfMsg : Platform.Router msg SelfMsg -> SelfMsg -> (State msg) -> Task Never (State msg)\nonSelfMsg _ _ state =\n    Task.succeed state\n\n\n{-| Subscribe to incoming HTTP requests.\n-}\nonRequest : Server -> (Request -> Response -> msg) -> Sub msg\nonRequest server tagger =\n    subscription (OnRequestSub server tagger)\n",
        "module Process exposing (Id, spawn, sleep, kill)\n\n{-| A process is essentially a function running concurrently to your application.\n\n\n@docs Id, spawn, sleep, kill\n\n\n## Future Plans\n\nRight now, this library is pretty sparse. For example, there is no public API\nfor processes to communicate with each other. This is a really important\nability, but it is also something that is extraordinarily easy to get wrong!\n\nI think the trend will be towards an Erlang style of concurrency, where every\nprocess has an “event queue” that anyone can send messages to. I currently\nthink the API will be extended to be more like this:\n\n    type Id exit msg\n\n    spawn : Task exit a -> Task x (Id exit Never)\n\n    kill : Id exit msg -> Task x {}\n\n    send : Id exit msg -> msg -> Task x {}\n\nA process `Id` will have two type variables to make sure all communication is\nvalid. The `exit` type describes the messages that are produced if the process\nfails because of user code. So if processes are linked and trapping errors,\nthey will need to handle this. The `msg` type just describes what kind of\nmessages this process can be sent by strangers.\n\nWe shall see though! This is just a draft that does not cover nearly everything\nit needs to, so the long-term vision for concurrency in Gren will be rolling out\nslowly as I get more data and experience.\n\nI ask that people bullish on compiling to node.js keep this in mind. I think we\ncan do better than the hopelessly bad concurrency model of node.js, and I hope\nthe Gren community will be supportive of being more ambitious, even if it takes\nlonger. That’s kind of what Gren is all about.\n\n-}\n\nimport Basics exposing (Float, Never)\nimport Gren.Kernel.Process\nimport Gren.Kernel.Scheduler\nimport Platform\nimport Task exposing (Task)\n\n\n{-| A light-weight process that runs concurrently. You can use `spawn` to\nget a bunch of different tasks running in different processes. The Gren runtime\nwill interleave their progress. So if a task is taking too long, we will pause\nit at an `andThen` and switch over to other stuff.\n\n**Note:** We make a distinction between _concurrency_ which means interleaving\ndifferent sequences and _parallelism_ which means running different\nsequences at the exact same time. For example, a\n[time-sharing system](https://en.wikipedia.org/wiki/Time-sharing) is definitely\nconcurrent, but not necessarily parallel. So even though JS runs within a\nsingle OS-level thread, Gren can still run things concurrently.\n\n-}\ntype alias Id =\n    Platform.ProcessId\n\n\n{-| Run a task in its own light-weight process. In the following example,\n`task1` and `task2` will be interleaved. If `task1` makes a long HTTP request\nor is just taking a long time, we can hop over to `task2` and do some work\nthere.\n\n    spawn task1\n        |> Task.andThen (\\_ -> spawn task2)\n\n**Note:** This creates a relatively restricted kind of `Process` because it\ncannot receive any messages. More flexibility for user-defined processes will\ncome in a later release!\n\n-}\nspawn : Task x a -> Task y Id\nspawn =\n    Gren.Kernel.Scheduler.spawn\n\n\n{-| Block progress on the current process for the given number of milliseconds.\nThe JavaScript equivalent of this is [`setTimeout`][setTimeout] which lets you\ndelay work until later.\n\n[setTimeout]: https://developer.mozilla.org/en-US/docs/Web/API/WindowTimers/setTimeout\n\n-}\nsleep : Float -> Task x {}\nsleep =\n    Gren.Kernel.Process.sleep\n\n\n{-| Sometimes you `spawn` a process, but later decide it would be a waste to\nhave it keep running and doing stuff. The `kill` function will force a process\nto bail on whatever task it is running. So if there is an HTTP request in\nflight, it will also abort the request.\n-}\nkill : Id -> Task x {}\nkill =\n    Gren.Kernel.Scheduler.kill\n"
    ],
    "names": [
        "Task.andThen",
        "_Scheduler_andThen",
        "Basics.apL",
        "f",
        "x",
        "Basics.apR",
        "func",
        "acc",
        "dict",
        "key",
        "value",
        "Dict.foldl",
        "left",
        "right",
        "Array.pushLast",
        "_Array_push",
        "Dict.keys",
        "keyArray",
        "Set.toArray",
        "_v0",
        "Basics.add",
        "_Basics_add",
        "String.all",
        "_String_all",
        "Basics.and",
        "_Basics_and",
        "Basics.append",
        "_Utils_append",
        "Json.Encode.encode",
        "_Json_encode",
        "String.fromInt",
        "_String_fromNumber",
        "String.join",
        "_String_join",
        "String.split",
        "_String_split",
        "Json.Decode.indent",
        "str",
        "Array.indexedMap",
        "_Array_indexedMap",
        "Basics.le",
        "_Utils_le",
        "Char.toCode",
        "_Char_toCode",
        "Char.isLower",
        "_char",
        "code",
        "char",
        "Char.isUpper",
        "Basics.or",
        "_Basics_or",
        "Char.isAlpha",
        "Char.isDigit",
        "Char.isAlphaNum",
        "Array.length",
        "_Array_length",
        "String.uncons",
        "_String_uncons",
        "i",
        "error",
        "Json.Decode.errorToString",
        "Json.Decode.errorToStringHelp",
        "context",
        "isSimple",
        "_v1",
        "rest",
        "fieldName",
        "err",
        "indexName",
        "starter",
        "introduction",
        "errors",
        "Json.Decode.errorOneOf",
        "json",
        "msg",
        "Result.isOk",
        "result",
        "Dict.empty",
        "Dict.RBEmpty_gren_builtin",
        "Dict.balance",
        "color",
        "Dict.Red",
        "Dict.Black",
        "lK",
        "lV",
        "lLeft",
        "lRight",
        "rK",
        "rV",
        "rLeft",
        "rRight",
        "llK",
        "llV",
        "llLeft",
        "llRight",
        "Basics.compare",
        "_Utils_compare",
        "nKey",
        "nColor",
        "nValue",
        "Dict.setHelp",
        "nLeft",
        "nRight",
        "Dict.set",
        "k",
        "v",
        "l",
        "r",
        "String.toLower",
        "_String_toLower",
        "Node.archFromString",
        "arch",
        "Node.Arm",
        "Node.Arm64",
        "Node.IA32",
        "Node.Mips",
        "Node.Mipsel",
        "Node.PPC",
        "Node.PPC64",
        "Node.S390",
        "Node.S390x",
        "Node.X64",
        "Node.UnknownArchitecture",
        "Task.succeed",
        "_Scheduler_succeed",
        "Task.map",
        "taskA",
        "a",
        "Node.platformFromString",
        "platform",
        "Node.Win32",
        "Node.Darwin",
        "Node.Linux",
        "Node.FreeBSD",
        "Node.OpenBSD",
        "Node.SunOS",
        "Node.Aix",
        "Node.UnknownPlatform",
        "Node.initializeEnvironment",
        "raw",
        "applicationPath",
        "args",
        "cpuArchitecture",
        "stderr",
        "stdin",
        "stdout",
        "_Node_init",
        "Task.init",
        "Array.map",
        "_Array_map",
        "Array.foldr",
        "_Array_foldr",
        "Task.map2",
        "taskB",
        "b",
        "Array.prepend",
        "_Array_append",
        "Array.pushFirst",
        "array",
        "Task.sequence",
        "tasks",
        "Platform.sendToApp",
        "_Platform_sendToApp",
        "Task.spawnCmd",
        "router",
        "cmd",
        "_Scheduler_spawn",
        "task",
        "Task.onEffects",
        "commands",
        "state",
        "Task.onSelfMsg",
        "_v2",
        "Task.cmdMap",
        "tagger",
        "Task.Perform",
        "Task.Execute",
        "Task.perform",
        "toMessage",
        "Task.command",
        "Node.unwrap",
        "Node.init",
        "initTask",
        "command",
        "Node.InitDone",
        "env",
        "model",
        "Node.Uninitialized",
        "Platform.Sub.map",
        "_Platform_map",
        "Platform.Sub.batch",
        "_Platform_batch",
        "Platform.Sub.none",
        "Node.subscriptions",
        "appSubs",
        "Node.MsgReceived",
        "appModel",
        "Platform.Cmd.map",
        "Platform.Cmd.batch",
        "Platform.Cmd.none",
        "Node.update",
        "appUpdate",
        "initResult",
        "Node.Initialized",
        "updateResult",
        "appMsg",
        "Platform.worker",
        "_Platform_worker",
        "Node.defineProgram",
        "config",
        "init",
        "subscriptions",
        "update",
        "Basics.composeL",
        "g",
        "Task.onError",
        "_Scheduler_onError",
        "Task.attempt",
        "resultToMessage",
        "Result.Err",
        "Result.Ok",
        "Basics.identity",
        "Init.unwrap",
        "Init.await",
        "fn",
        "Internal.Init.Task",
        "Init.awaitTask",
        "Time.millisToPosix",
        "Time.Posix",
        "FileSystem.checkAccess",
        "permissions",
        "path",
        "_FileSystem_access",
        "Basics.sub",
        "_Basics_sub",
        "n",
        "rec",
        "Array.slice",
        "_Array_slice",
        "Array.dropFirst",
        "FileSystem.Path.empty",
        "directory",
        "extension",
        "filename",
        "root",
        "Basics.eq",
        "_Utils_equal",
        "Task.execute",
        "FileSystem.Path.fromPosixString",
        "_FilePath_fromPosix",
        "FileSystem.Path.fromWin32String",
        "_FilePath_fromWin32",
        "targetKey",
        "Maybe.Nothing",
        "Maybe.Just",
        "Node.getEnvironmentVariables",
        "_Node_getEnvironmentVariables",
        "FileSystem.homeDirectory",
        "_FileSystem_homeDir",
        "ChildProcess.initialize",
        "ChildProcess.Permission",
        "FileSystem.initialize",
        "FileSystem.Permission",
        "HttpClient.initialize",
        "HttpClient.AnyPermission",
        "Terminal.initialize",
        "isTTY",
        "colorDepth",
        "columns",
        "permission",
        "Terminal.Permission",
        "rows",
        "_Terminal_init",
        "Array.append",
        "fst",
        "second",
        "String.isEmpty",
        "string",
        "FileSystem.Path.filenameWithExtension",
        "Array.filter",
        "_Array_filter",
        "Basics.neq",
        "_Utils_notEqual",
        "FileSystem.Path.prepend",
        "dir",
        "FileSystem.Path.append",
        "Main.compilerVersion",
        "Maybe.map",
        "maybe",
        "Maybe.withDefault",
        "_default",
        "default",
        "Main.makeLocalPath",
        "homeDir",
        "envVars",
        "startPath",
        "Dict.get",
        "endPath",
        "Main.makeRemotePath",
        "Bytes.Encode.getLength",
        "builder",
        "w",
        "_Bytes_length",
        "bs",
        "Array.foldl",
        "_Array_foldl",
        "mb",
        "offset",
        "_Bytes_write_i8",
        "_Bytes_write_i16",
        "e",
        "Bytes.LE",
        "_Bytes_write_i32",
        "_Bytes_write_u8",
        "_Bytes_write_u16",
        "_Bytes_write_u32",
        "_Bytes_write_f32",
        "_Bytes_write_f64",
        "Bytes.Encode.writeSequence",
        "_Bytes_write_string",
        "s",
        "_Bytes_write_bytes",
        "builders",
        "currentOffset",
        "Bytes.Encode.write",
        "Bytes.fromString",
        "_Bytes_fromString",
        "Stream.send",
        "bytes",
        "_Stream_send",
        "kernelStream",
        "Stream.sendString",
        "stream",
        "Stream.sendLine",
        "Node.startProgram",
        "FileSystem.Path.toPosixString",
        "_FilePath_toPosix",
        "FileSystem.Path.toWin32String",
        "_FilePath_toWin32",
        "Main.init",
        "fsPermission",
        "cpPermission",
        "httpPermission",
        "terminalConfig",
        "userArgs",
        "useless",
        "Main.countDown",
        "useColor",
        "_v10",
        "maybePaths",
        "override",
        "localPath",
        "overridePath",
        "remotePath",
        "paths",
        "pathToString",
        "Main.ExistanceChecked",
        "Json.Decode.succeed",
        "_Json_succeed",
        "Array.findFirst",
        "_Array_findFirst",
        "Array.member",
        "FileSystem.accessPermissionsToInt",
        "values",
        "numberFor",
        "num",
        "FileSystem.Read",
        "FileSystem.Write",
        "FileSystem.Execute",
        "FileSystem.changeAccess",
        "mode",
        "owner",
        "group",
        "others",
        "_FileSystem_chmod",
        "HttpClient.expectBytes",
        "req",
        "body",
        "expect",
        "HttpClient.ExpectBytes",
        "headers",
        "method",
        "timeout",
        "url",
        "Basics.mul",
        "_Basics_mul",
        "HttpClient.defaultTimeout",
        "HttpClient.request",
        "HttpClient.BodyEmpty",
        "HttpClient.ExpectAnything",
        "HttpClient.get",
        "HttpServer.GET",
        "Json.Decode.decodeString",
        "_Json_runOnString",
        "HttpClient.bodyTypeAsString",
        "HttpClient.expectTypeAsString",
        "HttpServer.methodToString",
        "HttpClient.kernelRequestConfig",
        "actualUrl",
        "prefix",
        "bodyType",
        "expectType",
        "HttpClient.send",
        "_HttpClient_request",
        "Main.downloadBinary",
        "FileSystem.errorToString",
        "message",
        "HttpClient.errorToString",
        "res",
        "statusCode",
        "statusText",
        "debugStr",
        "FileSystem.makeDirectory",
        "options",
        "_FileSystem_makeDirectory",
        "Array.dropLast",
        "Array.get",
        "_Array_get",
        "Basics.negate",
        "Array.last",
        "Array.popLast",
        "initial",
        "last",
        "FileSystem.Path.parentPath",
        "_v3",
        "ext",
        "file",
        "ChildProcess.defaultSpawnOptions",
        "connection",
        "ChildProcess.Integrated",
        "environmentVariables",
        "ChildProcess.InheritEnvironmentVariables",
        "runDuration",
        "ChildProcess.NoLimit",
        "shell",
        "ChildProcess.DefaultShell",
        "workingDirectory",
        "ChildProcess.InheritWorkingDirectory",
        "Dict.singleton",
        "Basics.gt",
        "_Utils_gt",
        "Basics.max",
        "y",
        "Process.spawn",
        "ChildProcess.spawn",
        "program",
        "_arguments",
        "opts",
        "_ChildProcess_spawn",
        "arguments",
        "option",
        "ms",
        "_v4",
        "choice",
        "_v5",
        "inherit",
        "Main.runCompiler",
        "colorEnvVar",
        "ChildProcess.MergeWithEnvironmentVariables",
        "FileSystem.writeFile",
        "_FileSystem_writeFile",
        "Main.update",
        "Main.CompilerDownloaded",
        "location",
        "cacheFolder",
        "Main.CompilerInstalled",
        "data",
        "recursive",
        "fsErr",
        "Main.main"
    ],
    "mappings": "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"
} \ No newline at end of file diff --git a/src/Main.gren b/src/Main.gren index d0fdb80e..27d0d53f 100644 --- a/src/Main.gren +++ b/src/Main.gren @@ -41,13 +41,6 @@ compilerVersion : String compilerVersion = "0.4.4" -countDown : Int -> {} -> Int -countDown n rec = - if n <= 0 then - 0 - - else - countDown (n - 1) rec init : Node.Environment -> Init.Task { model : Model, command : Cmd Msg } init env = @@ -61,8 +54,6 @@ init env = userArgs = Array.dropFirst 2 env.args - useless = countDown 10 {} - useColor = case terminalConfig of Nothing ->