/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

  1. %%% Copyright (C) 2008 - Will Glozer. All rights reserved.
  2. %% Modified by Marc Worrell, 2009.
  3. %% Added last_id/2, reset_id/2, squery1/2, equery1/2, equery1/3, assoc/2, assoc/3, columns/2
  4. %% Adapted with_transaction/2
  5. -module(pgsql).
  6. -export([connect/2, connect/3, connect/4, close/1]).
  7. -export([last_id/2, reset_id/2, squery1/2, equery1/2, equery1/3, assoc/2, assoc/3]).
  8. -export([get_parameter/2, squery/2, equery/2, equery/3]).
  9. -export([parse/2, parse/3, parse/4, describe/2, describe/3]).
  10. -export([bind/3, bind/4, execute/2, execute/3, execute/4]).
  11. -export([close/2, close/3, sync/1]).
  12. -export([with_transaction/2]).
  13. -include("pgsql.hrl").
  14. -include("zotonic.hrl").
  15. -define(timeout, 5000).
  16. %% -- client interface --
  17. connect(Host, Opts) ->
  18. connect(Host, os:getenv("USER"), "", Opts).
  19. connect(Host, Username, Opts) ->
  20. connect(Host, Username, "", Opts).
  21. connect(Host, Username, Password, Opts) ->
  22. {ok, C} = pgsql_connection:start_link(),
  23. pgsql_connection:connect(C, Host, Username, Password, Opts).
  24. close(C) when is_pid(C) ->
  25. catch pgsql_connection:stop(C),
  26. ok.
  27. get_parameter(C, Name) ->
  28. pgsql_connection:get_parameter(C, Name).
  29. last_id(C, Table) when is_atom(Table) ->
  30. last_id(C, atom_to_list(Table));
  31. last_id(C, Table) ->
  32. equery1(C, "select currval(pg_get_serial_sequence($1, 'id'))", [Table]).
  33. reset_id(C, Table) when is_atom(Table) ->
  34. reset_id(C, atom_to_list(Table));
  35. reset_id(C, Table) ->
  36. {ok, Max} = equery1(C, "select max(id) from \""++Table++"\""),
  37. equery1(C, "select setval(pg_get_serial_sequence($1, 'id'), $2)", [Table, Max+1]).
  38. assoc(C, Sql) ->
  39. assoc(C, Sql, []).
  40. assoc(C, Sql, Parameters) ->
  41. case equery(C, Sql, Parameters) of
  42. {ok, Columns, Rows} ->
  43. Names = [ list_to_atom(binary_to_list(Name)) || #column{name=Name} <- Columns ],
  44. Rows1 = [ lists:zip(Names, tuple_to_list(Row)) || Row <- Rows ],
  45. {ok, Rows1};
  46. Other -> Other
  47. end.
  48. squery1(C, Sql) ->
  49. case squery(C,Sql) of
  50. {ok, _Columns, []} -> {error, noresult};
  51. {ok, _RowCount, _Columns, []} -> {error, noresult};
  52. {ok, _Columns, [Row|_]} -> {ok, element(1, Row)};
  53. {ok, _RowCount, _Columns, [Row|_]} -> {ok, element(1, Row)};
  54. Other -> Other
  55. end.
  56. equery1(C, Sql) ->
  57. equery1(C, Sql, []).
  58. equery1(C, Sql, Parameters) ->
  59. case equery(C, Sql, Parameters) of
  60. {ok, _Columns, []} -> {error, noresult};
  61. {ok, _RowCount, _Columns, []} -> {error, noresult};
  62. {ok, _Columns, [Row|_]} -> {ok, element(1, Row)};
  63. {ok, _RowCount, _Columns, [Row|_]} -> {ok, element(1, Row)};
  64. Other -> Other
  65. end.
  66. squery(C, Sql) ->
  67. ok = pgsql_connection:squery(C, Sql),
  68. case receive_results(C, []) of
  69. [Result] -> Result;
  70. Results -> Results
  71. end.
  72. equery(C, Sql) ->
  73. equery(C, Sql, []).
  74. equery(C, Sql, Parameters) when is_tuple(Parameters) ->
  75. equery(C, Sql, tuple_to_list(Parameters));
  76. equery(C, Sql, Parameters) ->
  77. case pgsql_connection:parse(C, "", Sql, []) of
  78. {ok, #statement{types = Types} = S} ->
  79. Typed_Parameters = lists:zip(Types, Parameters),
  80. ok = pgsql_connection:equery(C, S, Typed_Parameters),
  81. receive_result(C);
  82. Error ->
  83. ?LOG("SQL error ~p : ~p", [Error, Sql]),
  84. Error
  85. end.
  86. %% parse
  87. parse(C, Sql) ->
  88. parse(C, "", Sql, []).
  89. parse(C, Sql, Types) ->
  90. parse(C, "", Sql, Types).
  91. parse(C, Name, Sql, Types) ->
  92. pgsql_connection:parse(C, Name, Sql, Types).
  93. %% bind
  94. bind(C, Statement, Parameters) ->
  95. bind(C, Statement, "", Parameters).
  96. bind(C, Statement, PortalName, Parameters) ->
  97. pgsql_connection:bind(C, Statement, PortalName, Parameters).
  98. %% execute
  99. execute(C, S) ->
  100. execute(C, S, "", 0).
  101. execute(C, S, N) ->
  102. execute(C, S, "", N).
  103. execute(C, S, PortalName, N) ->
  104. pgsql_connection:execute(C, S, PortalName, N),
  105. receive_extended_result(C).
  106. %% statement/portal functions
  107. describe(C, #statement{name = Name}) ->
  108. pgsql_connection:describe(C, statement, Name).
  109. describe(C, Type, Name) ->
  110. pgsql_connection:describe(C, Type, Name).
  111. close(C, #statement{name = Name}) ->
  112. pgsql_connection:close(C, statement, Name).
  113. close(C, Type, Name) ->
  114. pgsql_connection:close(C, Type, Name).
  115. sync(C) ->
  116. pgsql_connection:sync(C).
  117. %% misc helper functions
  118. with_transaction(C, F) ->
  119. try
  120. {ok, [], []} = squery(C, "BEGIN"),
  121. R = F(C),
  122. {ok, [], []} = squery(C, "COMMIT"),
  123. R
  124. catch
  125. E:Why ->
  126. ?LOG("Exception in transaction: \"~p,~p\"", [E,Why]),
  127. squery(C, "ROLLBACK"),
  128. {rollback, Why}
  129. end.
  130. %% -- internal functions --
  131. receive_result(C) ->
  132. R = receive_result(C, [], []),
  133. receive
  134. {pgsql, C, done} -> R
  135. end.
  136. receive_results(C, Results) ->
  137. case receive_result(C, [], []) of
  138. done -> lists:reverse(Results);
  139. R -> receive_results(C, [R | Results])
  140. end.
  141. receive_result(C, Cols, Rows) ->
  142. receive
  143. {pgsql, C, {columns, Cols2}} ->
  144. receive_result(C, Cols2, Rows);
  145. {pgsql, C, {data, Row}} ->
  146. receive_result(C, Cols, [Row | Rows]);
  147. {pgsql, C, {error, _E} = Error} ->
  148. Error;
  149. {pgsql, C, {complete, {_Type, Count}}} ->
  150. case Rows of
  151. [] -> {ok, Count};
  152. _L -> {ok, Count, Cols, lists:reverse(Rows)}
  153. end;
  154. {pgsql, C, {complete, _Type}} ->
  155. {ok, Cols, lists:reverse(Rows)};
  156. {pgsql, C, {notice, _N}} ->
  157. receive_result(C, Cols, Rows);
  158. {pgsql, C, done} ->
  159. done
  160. after
  161. ?timeout -> {error, timeout}
  162. end.
  163. receive_extended_result(C)->
  164. receive_extended_result(C, []).
  165. receive_extended_result(C, Rows) ->
  166. receive
  167. {pgsql, C, {data, Row}} ->
  168. receive_extended_result(C, [Row | Rows]);
  169. {pgsql, C, {error, _E} = Error} ->
  170. Error;
  171. {pgsql, C, suspended} ->
  172. {partial, lists:reverse(Rows)};
  173. {pgsql, C, {complete, {_Type, Count}}} ->
  174. case Rows of
  175. [] -> {ok, Count};
  176. _L -> {ok, Count, lists:reverse(Rows)}
  177. end;
  178. {pgsql, C, {complete, _Type}} ->
  179. {ok, lists:reverse(Rows)};
  180. {pgsql, C, {notice, _N}} ->
  181. receive_extended_result(C, Rows)
  182. after
  183. ?timeout -> {error, timeout}
  184. end.