/src/dbdrivers/postgresql/epgsql/pgsql.erl
https://code.google.com/p/zotonic/ · Erlang · 230 lines · 173 code · 46 blank · 11 comment · 2 complexity · 1c1d18612b5714aa6b0d741553179a8c MD5 · raw file
- %%% Copyright (C) 2008 - Will Glozer. All rights reserved.
- %% Modified by Marc Worrell, 2009.
- %% Added last_id/2, reset_id/2, squery1/2, equery1/2, equery1/3, assoc/2, assoc/3, columns/2
- %% Adapted with_transaction/2
- -module(pgsql).
- -export([connect/2, connect/3, connect/4, close/1]).
- -export([last_id/2, reset_id/2, squery1/2, equery1/2, equery1/3, assoc/2, assoc/3]).
- -export([get_parameter/2, squery/2, equery/2, equery/3]).
- -export([parse/2, parse/3, parse/4, describe/2, describe/3]).
- -export([bind/3, bind/4, execute/2, execute/3, execute/4]).
- -export([close/2, close/3, sync/1]).
- -export([with_transaction/2]).
- -include("pgsql.hrl").
- -include("zotonic.hrl").
- -define(timeout, 5000).
- %% -- client interface --
- connect(Host, Opts) ->
- connect(Host, os:getenv("USER"), "", Opts).
- connect(Host, Username, Opts) ->
- connect(Host, Username, "", Opts).
- connect(Host, Username, Password, Opts) ->
- {ok, C} = pgsql_connection:start_link(),
- pgsql_connection:connect(C, Host, Username, Password, Opts).
- close(C) when is_pid(C) ->
- catch pgsql_connection:stop(C),
- ok.
- get_parameter(C, Name) ->
- pgsql_connection:get_parameter(C, Name).
- last_id(C, Table) when is_atom(Table) ->
- last_id(C, atom_to_list(Table));
- last_id(C, Table) ->
- equery1(C, "select currval(pg_get_serial_sequence($1, 'id'))", [Table]).
- reset_id(C, Table) when is_atom(Table) ->
- reset_id(C, atom_to_list(Table));
- reset_id(C, Table) ->
- {ok, Max} = equery1(C, "select max(id) from \""++Table++"\""),
- equery1(C, "select setval(pg_get_serial_sequence($1, 'id'), $2)", [Table, Max+1]).
- assoc(C, Sql) ->
- assoc(C, Sql, []).
- assoc(C, Sql, Parameters) ->
- case equery(C, Sql, Parameters) of
- {ok, Columns, Rows} ->
- Names = [ list_to_atom(binary_to_list(Name)) || #column{name=Name} <- Columns ],
- Rows1 = [ lists:zip(Names, tuple_to_list(Row)) || Row <- Rows ],
- {ok, Rows1};
- Other -> Other
- end.
- squery1(C, Sql) ->
- case squery(C,Sql) of
- {ok, _Columns, []} -> {error, noresult};
- {ok, _RowCount, _Columns, []} -> {error, noresult};
- {ok, _Columns, [Row|_]} -> {ok, element(1, Row)};
- {ok, _RowCount, _Columns, [Row|_]} -> {ok, element(1, Row)};
- Other -> Other
- end.
- equery1(C, Sql) ->
- equery1(C, Sql, []).
- equery1(C, Sql, Parameters) ->
- case equery(C, Sql, Parameters) of
- {ok, _Columns, []} -> {error, noresult};
- {ok, _RowCount, _Columns, []} -> {error, noresult};
- {ok, _Columns, [Row|_]} -> {ok, element(1, Row)};
- {ok, _RowCount, _Columns, [Row|_]} -> {ok, element(1, Row)};
- Other -> Other
- end.
-
- squery(C, Sql) ->
- ok = pgsql_connection:squery(C, Sql),
- case receive_results(C, []) of
- [Result] -> Result;
- Results -> Results
- end.
- equery(C, Sql) ->
- equery(C, Sql, []).
- equery(C, Sql, Parameters) when is_tuple(Parameters) ->
- equery(C, Sql, tuple_to_list(Parameters));
- equery(C, Sql, Parameters) ->
- case pgsql_connection:parse(C, "", Sql, []) of
- {ok, #statement{types = Types} = S} ->
- Typed_Parameters = lists:zip(Types, Parameters),
- ok = pgsql_connection:equery(C, S, Typed_Parameters),
- receive_result(C);
- Error ->
- ?LOG("SQL error ~p : ~p", [Error, Sql]),
- Error
- end.
- %% parse
- parse(C, Sql) ->
- parse(C, "", Sql, []).
- parse(C, Sql, Types) ->
- parse(C, "", Sql, Types).
- parse(C, Name, Sql, Types) ->
- pgsql_connection:parse(C, Name, Sql, Types).
- %% bind
- bind(C, Statement, Parameters) ->
- bind(C, Statement, "", Parameters).
- bind(C, Statement, PortalName, Parameters) ->
- pgsql_connection:bind(C, Statement, PortalName, Parameters).
- %% execute
- execute(C, S) ->
- execute(C, S, "", 0).
- execute(C, S, N) ->
- execute(C, S, "", N).
- execute(C, S, PortalName, N) ->
- pgsql_connection:execute(C, S, PortalName, N),
- receive_extended_result(C).
- %% statement/portal functions
- describe(C, #statement{name = Name}) ->
- pgsql_connection:describe(C, statement, Name).
- describe(C, Type, Name) ->
- pgsql_connection:describe(C, Type, Name).
- close(C, #statement{name = Name}) ->
- pgsql_connection:close(C, statement, Name).
- close(C, Type, Name) ->
- pgsql_connection:close(C, Type, Name).
- sync(C) ->
- pgsql_connection:sync(C).
- %% misc helper functions
- with_transaction(C, F) ->
- try
- {ok, [], []} = squery(C, "BEGIN"),
- R = F(C),
- {ok, [], []} = squery(C, "COMMIT"),
- R
- catch
- E:Why ->
- ?LOG("Exception in transaction: \"~p,~p\"", [E,Why]),
- squery(C, "ROLLBACK"),
- {rollback, Why}
- end.
- %% -- internal functions --
- receive_result(C) ->
- R = receive_result(C, [], []),
- receive
- {pgsql, C, done} -> R
- end.
- receive_results(C, Results) ->
- case receive_result(C, [], []) of
- done -> lists:reverse(Results);
- R -> receive_results(C, [R | Results])
- end.
- receive_result(C, Cols, Rows) ->
- receive
- {pgsql, C, {columns, Cols2}} ->
- receive_result(C, Cols2, Rows);
- {pgsql, C, {data, Row}} ->
- receive_result(C, Cols, [Row | Rows]);
- {pgsql, C, {error, _E} = Error} ->
- Error;
- {pgsql, C, {complete, {_Type, Count}}} ->
- case Rows of
- [] -> {ok, Count};
- _L -> {ok, Count, Cols, lists:reverse(Rows)}
- end;
- {pgsql, C, {complete, _Type}} ->
- {ok, Cols, lists:reverse(Rows)};
- {pgsql, C, {notice, _N}} ->
- receive_result(C, Cols, Rows);
- {pgsql, C, done} ->
- done
- after
- ?timeout -> {error, timeout}
- end.
- receive_extended_result(C)->
- receive_extended_result(C, []).
- receive_extended_result(C, Rows) ->
- receive
- {pgsql, C, {data, Row}} ->
- receive_extended_result(C, [Row | Rows]);
- {pgsql, C, {error, _E} = Error} ->
- Error;
- {pgsql, C, suspended} ->
- {partial, lists:reverse(Rows)};
- {pgsql, C, {complete, {_Type, Count}}} ->
- case Rows of
- [] -> {ok, Count};
- _L -> {ok, Count, lists:reverse(Rows)}
- end;
- {pgsql, C, {complete, _Type}} ->
- {ok, lists:reverse(Rows)};
- {pgsql, C, {notice, _N}} ->
- receive_extended_result(C, Rows)
- after
- ?timeout -> {error, timeout}
- end.