diff options
| author | Joel Martin <github@martintribe.org> | 2014-03-24 16:32:24 -0500 |
|---|---|---|
| committer | Joel Martin <github@martintribe.org> | 2014-03-24 16:32:24 -0500 |
| commit | 3169070063b2cb877200117ebb384269d73bcb93 (patch) | |
| tree | 23de3db1ea5c37afd21a45b6ed7771f56a08c0c4 /bash/stepA_more.sh | |
| download | mal-3169070063b2cb877200117ebb384269d73bcb93.tar.gz mal-3169070063b2cb877200117ebb384269d73bcb93.zip | |
Current state of mal for Clojure West lighting talk.
Diffstat (limited to 'bash/stepA_more.sh')
| -rwxr-xr-x | bash/stepA_more.sh | 282 |
1 files changed, 282 insertions, 0 deletions
diff --git a/bash/stepA_more.sh b/bash/stepA_more.sh new file mode 100755 index 0000000..8caa72d --- /dev/null +++ b/bash/stepA_more.sh @@ -0,0 +1,282 @@ +#!/bin/bash + +INTERACTIVE=${INTERACTIVE-yes} + +source $(dirname $0)/reader.sh + +# READ: read and parse input +READ () { + READLINE + READ_STR "${r}" +} + +IS_PAIR () { + if _list? "${1}"; then + _count "${1}" + [[ "${r}" > 0 ]] && return 0 + fi + return 1 +} + +QUASIQUOTE () { + if ! IS_PAIR "${1}"; then + symbol quote + list "${r}" "${1}" + return + else + _nth "${1}" 0; local a0="${r}" + if [[ "${ANON["${a0}"]}" == "unquote" ]]; then + _nth "${1}" 1 + return + elif IS_PAIR "${a0}"; then + _nth "${a0}" 0; local a00="${r}" + if [[ "${ANON["${a00}"]}" == "splice-unquote" ]]; then + symbol concat; local a="${r}" + _nth "${a0}" 1; local b="${r}" + rest "${1}" + QUASIQUOTE "${r}"; local c="${r}" + list "${a}" "${b}" "${c}" + return + fi + fi + fi + symbol cons; local a="${r}" + QUASIQUOTE "${a0}"; local b="${r}" + rest "${1}" + QUASIQUOTE "${r}"; local c="${r}" + list "${a}" "${b}" "${c}" + return +} + +IS_MACRO_CALL () { + if ! _list? "${1}"; then return 1; fi + _nth "${1}" 0; local a0="${r}" + if _symbol? "${a0}"; then + ENV_FIND "${2}" "${ANON["${a0}"]}_ismacro_" + if [[ "${r}" ]]; then + return 0 + fi + fi + return 1 +} + +MACROEXPAND () { + local ast="${1}" env="${2}" + while IS_MACRO_CALL "${ast}" "${env}"; do + _nth "${ast}" 0; local a0="${r}" + ENV_GET "${env}" "${ANON["${a0}"]}"; local mac="${ANON["${r}"]}" + rest "${ast}" + ${mac%%@*} ${ANON["${r}"]} + ast="${r}" + done + r="${ast}" +} + + +EVAL_AST () { + local ast="${1}" env="${2}" + #_pr_str "${ast}"; echo "EVAL_AST '${ast}:${r} / ${env}'" + _obj_type "${ast}"; local ot="${r}" + case "${ot}" in + symbol) + local val="${ANON["${ast}"]}" + ENV_GET "${env}" "${val}" + return ;; + list) + _map_with_type list EVAL "${ast}" "${env}" ;; + vector) + _map_with_type vector EVAL "${ast}" "${env}" ;; + hash_map) + local res="" val="" hm="${ANON["${ast}"]}" + hash_map; local new_hm="${r}" + eval local keys="\${!${hm}[@]}" + for key in ${keys}; do + eval val="\${${hm}[\"${key}\"]}" + EVAL "${val}" "${env}" + assoc! "${new_hm}" "${key}" "${r}" + done + r="${new_hm}" ;; + *) + r="${ast}" ;; + esac +} + +# EVAL: evaluate the parameter +EVAL () { + local ast="${1}" env="${2}" + while true; do + r= + [[ "${__ERROR}" ]] && return 1 + #_pr_str "${ast}"; echo "EVAL '${r} / ${env}'" + if ! _list? "${ast}"; then + EVAL_AST "${ast}" "${env}" + return + fi + + # apply list + MACROEXPAND "${ast}" "${env}" + ast="${r}" + if ! _list? "${ast}"; then return; fi + _nth "${ast}" 0; local a0="${r}" + _nth "${ast}" 1; local a1="${r}" + _nth "${ast}" 2; local a2="${r}" + case "${ANON["${a0}"]}" in + def!) local k="${ANON["${a1}"]}" + #echo "def! ${k} to ${a2} in ${env}" + EVAL "${a2}" "${env}" + ENV_SET "${env}" "${k}" "${r}" + return ;; + let*) ENV "${env}"; local let_env="${r}" + local let_pairs=(${ANON["${a1}"]}) + local idx=0 + #echo "let: [${let_pairs[*]}] for ${a2}" + while [[ "${let_pairs["${idx}"]}" ]]; do + EVAL "${let_pairs[$(( idx + 1))]}" "${let_env}" + ENV_SET "${let_env}" "${ANON["${let_pairs[${idx}]}"]}" "${r}" + idx=$(( idx + 2)) + done + EVAL "${a2}" "${let_env}" + return ;; + quote) + r="${a1}" + return ;; + quasiquote) + QUASIQUOTE "${a1}" + EVAL "${r}" "${env}" + return ;; + defmacro!) + local k="${ANON["${a1}"]}" + EVAL "${a2}" "${env}" + ENV_SET "${env}" "${k}" "${r}" + ENV_SET "${env}" "${k}_ismacro_" "yes" + return ;; + macroexpand) + MACROEXPAND "${a1}" "${env}" + return ;; + sh*) MACROEXPAND "${a1}" "${env}" + EVAL "${r}" "${env}" + local output="" + local line="" + while read line; do + output="${output}${line}\n" + done < <(eval ${ANON["${r}"]}) + string "${output}" + return ;; + try*) MACROEXPAND "${a1}" "${env}" + EVAL "${r}" "${env}" + [[ -z "${__ERROR}" ]] && return + _nth "${a2}" 0; local a20="${r}" + if [ "${ANON["${a20}"]}" == "catch__STAR__" ]; then + _nth "${a2}" 1; local a21="${r}" + _nth "${a2}" 2; local a22="${r}" + list "${a21}"; local binds="${r}" + ENV "${env}" "${binds}" "${__ERROR}" + local try_env="${r}" + __ERROR= + MACROEXPAND "${a22}" "${try_env}" + EVAL "${r}" "${try_env}" + fi # if no catch* clause, just propagate __ERROR + return ;; + do) _count "${ast}" + _slice "${ast}" 1 $(( ${r} - 2 )) + EVAL_AST "${r}" "${env}" + [[ "${__ERROR}" ]] && r= && return 1 + last "${ast}" + ast="${r}" + # Continue loop + ;; + if) EVAL "${a1}" "${env}" + if [[ "${r}" == "${__false}" || "${r}" == "${__nil}" ]]; then + # eval false form + _nth "${ast}" 3; local a3="${r}" + if [[ "${a3}" ]]; then + ast="${a3}" + else + r="${__nil}" + return + fi + else + # eval true condition + ast="${a2}" + fi + # Continue loop + ;; + fn*) new_function "ENV \"${env}\" \"${a1}\" \"\${@}\"; \ + EVAL \"${a2}\" \"\${r}\"" \ + "${a2}" "${env}" "${a1}" + return ;; + *) EVAL_AST "${ast}" "${env}" + [[ "${__ERROR}" ]] && r= && return 1 + local el="${r}" + first "${el}"; local f="${ANON["${r}"]}" + rest "${el}"; local args="${ANON["${r}"]}" + #echo "invoke: [${f}] ${args}" + if [[ "${f//@/ }" != "${f}" ]]; then + set -- ${f//@/ } + ast="${2}" + ENV "${3}" "${4}" ${args} + env="${r}" + else + eval ${f%%@*} ${args} + return + fi + # Continue loop + ;; + esac + done +} +# PRINT: +PRINT () { + if [[ "${__ERROR}" ]]; then + _pr_str "${__ERROR}" yes + r="Error: ${r}" + __ERROR= + else + _pr_str "${1}" yes + fi +} + +# REPL: read, eval, print, loop +ENV; REPL_ENV="${r}" +REP () { + r= + READ_STR "${1}" + EVAL "${r}" "${REPL_ENV}" + PRINT "${r}" +} + +_fref () { new_function "${2} \"\${@}\""; ENV_SET "${REPL_ENV}" "${1}" "${r}"; } + +# Import types functions +for n in "${!types_ns[@]}"; do _fref "${n}" "${types_ns["${n}"]}"; done + +readline () { + READLINE "${ANON["${1}"]}" && string "${r}" || r="${__nil}"; +} +_fref "readline" readline +read_string () { READ_STR "${ANON["${1}"]}"; } +_fref "read-string" read_string +_eval () { + EVAL "${1}" "${REPL_ENV}" +} +_fref "eval" _eval +slurp () { string "$(cat "${ANON["${1}"]}")"; } +_fref "slurp" slurp +slurp_do () { string "(do $(cat "${ANON["${1}"]}"))"; } +_fref "slurp-do" slurp_do + +# Defined using the language itself +REP "(def! not (fn* (a) (if a false true)))" +REP "(defmacro! cond (fn* (& xs) (if (> (count xs) 0) (list 'if (first xs) (if (> (count xs) 1) (nth xs 1) (throw \"odd number of forms to cond\")) (cons 'cond (rest (rest xs)))))))" +REP "(defmacro! or (fn* (& xs) (if (empty? xs) nil (if (= 1 (count xs)) (first xs) \`(let* (or_FIXME ~(first xs)) (if or_FIXME or_FIXME (or ~@(rest xs))))))))" +REP "(def! load-file (fn* (f) (eval (read-string (slurp-do f)))))" + +if [[ "${1}" ]]; then + echo "${@}" + REP "(load-file \"${1}\")" && echo "${r}" +elif [[ -n "${INTERACTIVE}" ]]; then + while true; do + READLINE "user> " || exit "$?" + [[ "${r}" ]] && REP "${r}" && echo "${r}" + done +fi |
