Move implementations into impls/ dir
[jackhill/mal.git] / plpgsql / step4_if_fn_do.sql
diff --git a/plpgsql/step4_if_fn_do.sql b/plpgsql/step4_if_fn_do.sql
deleted file mode 100644 (file)
index 853be40..0000000
+++ /dev/null
@@ -1,207 +0,0 @@
-\i init.sql
-\i io.sql
-\i types.sql
-\i reader.sql
-\i printer.sql
-\i env.sql
-\i core.sql
-
--- ---------------------------------------------------------
--- step1_read_print.sql
-
--- read
-CREATE FUNCTION READ(line varchar)
-RETURNS integer AS $$
-BEGIN
-    RETURN read_str(line);
-END; $$ LANGUAGE plpgsql;
-
--- eval
-CREATE FUNCTION eval_ast(ast integer, env integer)
-RETURNS integer AS $$
-DECLARE
-    type           integer;
-    seq            integer[];
-    eseq           integer[];
-    hash           hstore;
-    ehash          hstore;
-    kv             RECORD;
-    e              integer;
-    result         integer;
-BEGIN
-    SELECT type_id INTO type FROM value WHERE value_id = ast;
-    CASE
-    WHEN type = 7 THEN
-    BEGIN
-        result := env_get(env, ast);
-    END;
-    WHEN type IN (8, 9) THEN
-    BEGIN
-        SELECT val_seq INTO seq FROM value WHERE value_id = ast;
-        -- Evaluate each entry creating a new sequence
-        FOR i IN 1 .. COALESCE(array_length(seq, 1), 0) LOOP
-            eseq[i] := EVAL(seq[i], env);
-        END LOOP;
-        INSERT INTO value (type_id, val_seq) VALUES (type, eseq)
-            RETURNING value_id INTO result;
-    END;
-    WHEN type = 10 THEN
-    BEGIN
-        SELECT val_hash INTO hash FROM value WHERE value_id = ast;
-        -- Evaluate each value for every key/value
-        FOR kv IN SELECT * FROM each(hash) LOOP
-            e := EVAL(CAST(kv.value AS integer), env);
-            IF ehash IS NULL THEN
-                ehash := hstore(kv.key, CAST(e AS varchar));
-            ELSE
-                ehash := ehash || hstore(kv.key, CAST(e AS varchar));
-            END IF;
-        END LOOP;
-        INSERT INTO value (type_id, val_hash) VALUES (type, ehash)
-            RETURNING value_id INTO result;
-    END;
-    ELSE
-        result := ast;
-    END CASE;
-
-    RETURN result;
-END; $$ LANGUAGE plpgsql;
-
-CREATE FUNCTION EVAL(ast integer, env integer)
-RETURNS integer AS $$
-DECLARE
-    type     integer;
-    a0       integer;
-    a0sym    varchar;
-    a1       integer;
-    let_env  integer;
-    idx      integer;
-    binds    integer[];
-    el       integer;
-    fn       integer;
-    fname    varchar;
-    args     integer[];
-    cond     integer;
-    fast     integer;
-    fparams  integer;
-    fenv     integer;
-    result   integer;
-BEGIN
-    --PERFORM writeline(format('EVAL: %s [%s]', pr_str(ast), ast));
-    SELECT type_id INTO type FROM value WHERE value_id = ast;
-    IF type <> 8 THEN
-        RETURN eval_ast(ast, env);
-    END IF;
-
-    a0 := _first(ast);
-    IF _symbol_Q(a0) THEN
-        a0sym := (SELECT val_string FROM value WHERE value_id = a0);
-    ELSE
-        a0sym := '__<*fn*>__';
-    END IF;
-
-    CASE
-    WHEN a0sym = 'def!' THEN
-    BEGIN
-        RETURN env_set(env, _nth(ast, 1), EVAL(_nth(ast, 2), env));
-    END;
-    WHEN a0sym = 'let*' THEN
-    BEGIN
-        let_env := env_new(env);
-        a1 := _nth(ast, 1);
-        binds := (SELECT val_seq FROM value WHERE value_id = a1);
-        idx := 1;
-        WHILE idx < array_length(binds, 1) LOOP
-            PERFORM env_set(let_env, binds[idx], EVAL(binds[idx+1], let_env));
-            idx := idx + 2;
-        END LOOP;
-        RETURN EVAL(_nth(ast, 2), let_env);
-    END;
-    WHEN a0sym = 'do' THEN
-    BEGIN
-        el := eval_ast(_rest(ast), env);
-        RETURN _nth(el, _count(el)-1);
-    END;
-    WHEN a0sym = 'if' THEN
-    BEGIN
-        cond := EVAL(_nth(ast, 1), env);
-        SELECT type_id INTO type FROM value WHERE value_id = cond;
-        IF type = 0 OR type = 1 THEN -- nil or false
-            IF _count(ast) > 3 THEN
-                RETURN EVAL(_nth(ast, 3), env);
-            ELSE
-                RETURN 0; -- nil
-            END IF;
-        ELSE
-            RETURN EVAL(_nth(ast, 2), env);
-        END IF;
-    END;
-    WHEN a0sym = 'fn*' THEN
-    BEGIN
-        RETURN _malfunc(_nth(ast, 2), _nth(ast, 1), env);
-    END;
-    ELSE
-    BEGIN
-        el := eval_ast(ast, env);
-        SELECT type_id, function_name, ast_id, params_id, env_id
-            INTO type, fname, fast, fparams, fenv
-            FROM value WHERE value_id = _first(el);
-        args := _restArray(el);
-        IF type = 11 THEN
-            EXECUTE format('SELECT %s($1);', fname)
-                INTO result USING args;
-            RETURN result;
-        ELSIF type = 12 THEN
-            RETURN EVAL(fast, env_new_bindings(fenv, fparams, args));
-        ELSE
-            RAISE EXCEPTION 'Invalid function call';
-        END IF;
-    END;
-    END CASE;
-END; $$ LANGUAGE plpgsql;
-
--- print
-CREATE FUNCTION PRINT(exp integer)
-RETURNS varchar AS $$
-BEGIN
-    RETURN pr_str(exp);
-END; $$ LANGUAGE plpgsql;
-
-
--- repl
-
--- repl_env is environment 0
-
-CREATE FUNCTION REP(line varchar)
-RETURNS varchar AS $$
-BEGIN
-    RETURN PRINT(EVAL(READ(line), 0));
-END; $$ LANGUAGE plpgsql;
-
--- core.sql: defined using SQL (in core.sql)
--- repl_env is created and populated with core functions in by core.sql
-
--- core.mal: defined using the language itself
-SELECT REP('(def! not (fn* (a) (if a false true)))') \g '/dev/null'
-
-CREATE FUNCTION MAIN_LOOP(pwd varchar)
-RETURNS integer AS $$
-DECLARE
-    line    varchar;
-    output  varchar;
-BEGIN
-    WHILE true
-    LOOP
-        BEGIN
-            line := readline('user> ', 0);
-            IF line IS NULL THEN RETURN 0; END IF;
-            IF line <> '' THEN
-                output := REP(line);
-                PERFORM writeline(output);
-            END IF;
-
-            EXCEPTION WHEN OTHERS THEN
-                PERFORM writeline('Error: ' || SQLERRM);
-        END;
-    END LOOP;
-END; $$ LANGUAGE plpgsql;