epgsql_cast.erl 4.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176
  1. %%% Copyright (C) 2008 - Will Glozer. All rights reserved.
  2. %%% Copyright (C) 2011 - Anton Lebedevich. All rights reserved.
  3. %%%
  4. %%% Emulates original epgsql API over epgsqla for original tests
  5. -module(epgsql_cast).
  6. -export([connect/1, connect/2, connect/3, connect/4, close/1]).
  7. -export([get_parameter/2, set_notice_receiver/2, get_cmd_status/1, squery/2, equery/2, equery/3]).
  8. -export([prepared_query/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, execute_batch/2]).
  11. -export([close/2, close/3, sync/1]).
  12. -export([with_transaction/2]).
  13. -export([receive_result/2, sync_on_error/2]).
  14. -include("epgsql.hrl").
  15. %% -- client interface --
  16. connect(Opts) ->
  17. Ref = epgsqla:connect(Opts),
  18. await_connect(Ref).
  19. connect(Host, Opts) ->
  20. Ref = epgsqla:connect(Host, Opts),
  21. await_connect(Ref).
  22. connect(Host, Username, Opts) ->
  23. Ref = epgsqla:connect(Host, Username, Opts),
  24. await_connect(Ref).
  25. connect(Host, Username, Password, Opts) ->
  26. Ref = epgsqla:connect(Host, Username, Password, Opts),
  27. %% TODO connect timeout
  28. await_connect(Ref).
  29. await_connect(Ref) ->
  30. receive
  31. {C, Ref, connected} ->
  32. {ok, C};
  33. {_C, Ref, Error = {error, _}} ->
  34. Error;
  35. {'EXIT', _C, _Reason} ->
  36. {error, closed}
  37. end.
  38. close(C) ->
  39. epgsqla:close(C).
  40. get_parameter(C, Name) ->
  41. epgsqla:get_parameter(C, Name).
  42. set_notice_receiver(C, PidOrName) ->
  43. epgsqla:set_notice_receiver(C, PidOrName).
  44. get_cmd_status(C) ->
  45. epgsqla:get_cmd_status(C).
  46. squery(C, Sql) ->
  47. Ref = epgsqla:squery(C, Sql),
  48. receive_result(C, Ref).
  49. equery(C, Sql) ->
  50. equery(C, Sql, []).
  51. %% TODO add fast_equery command that doesn't need parsed statement
  52. equery(C, Sql, Parameters) ->
  53. case parse(C, Sql) of
  54. {ok, #statement{types = Types} = S} ->
  55. Typed_Parameters = lists:zip(Types, Parameters),
  56. Ref = epgsqla:equery(C, S, Typed_Parameters),
  57. receive_result(C, Ref);
  58. Error ->
  59. Error
  60. end.
  61. prepared_query(C, Name, Parameters) ->
  62. case describe(C, statement, Name) of
  63. {ok, #statement{types = Types} = S} ->
  64. Typed_Parameters = lists:zip(Types, Parameters),
  65. Ref = epgsqla:prepared_query(C, S, Typed_Parameters),
  66. receive_result(C, Ref);
  67. Error ->
  68. Error
  69. end.
  70. %% parse
  71. parse(C, Sql) ->
  72. parse(C, "", Sql, []).
  73. parse(C, Sql, Types) ->
  74. parse(C, "", Sql, Types).
  75. parse(C, Name, Sql, Types) ->
  76. Ref = epgsqla:parse(C, Name, Sql, Types),
  77. sync_on_error(C, receive_result(C, Ref)).
  78. %% bind
  79. bind(C, Statement, Parameters) ->
  80. bind(C, Statement, "", Parameters).
  81. bind(C, Statement, PortalName, Parameters) ->
  82. Ref = epgsqla:bind(C, Statement, PortalName, Parameters),
  83. sync_on_error(C, receive_result(C, Ref)).
  84. %% execute
  85. execute(C, S) ->
  86. execute(C, S, "", 0).
  87. execute(C, S, N) ->
  88. execute(C, S, "", N).
  89. execute(C, S, PortalName, N) ->
  90. Ref = epgsqla:execute(C, S, PortalName, N),
  91. receive_result(C, Ref).
  92. execute_batch(C, Batch) ->
  93. Ref = epgsqla:execute_batch(C, Batch),
  94. receive_result(C, Ref).
  95. %% statement/portal functions
  96. describe(C, #statement{name = Name}) ->
  97. describe(C, statement, Name).
  98. describe(C, Type, Name) ->
  99. Ref = epgsqla:describe(C, Type, Name),
  100. %% TODO unknown result format of Describe portal
  101. sync_on_error(C, receive_result(C, Ref)).
  102. close(C, #statement{name = Name}) ->
  103. close(C, statement, Name).
  104. close(C, Type, Name) ->
  105. Ref = epgsqla:close(C, Type, Name),
  106. receive_result(C, Ref).
  107. sync(C) ->
  108. Ref = epgsqla:sync(C),
  109. receive_result(C, Ref).
  110. %% misc helper functions
  111. with_transaction(C, F) ->
  112. try {ok, [], []} = squery(C, "BEGIN"),
  113. R = F(C),
  114. {ok, [], []} = squery(C, "COMMIT"),
  115. R
  116. catch
  117. _:Why ->
  118. squery(C, "ROLLBACK"),
  119. %% TODO hides error stacktrace
  120. {rollback, Why}
  121. end.
  122. receive_result(C, Ref) ->
  123. %% TODO timeout
  124. receive
  125. {C, Ref, Result} ->
  126. Result;
  127. %% TODO no 'EXIT' for not linked processes
  128. {'EXIT', C, _Reason} ->
  129. {error, closed}
  130. end.
  131. sync_on_error(C, Error = {error, _}) ->
  132. Ref = epgsqla:sync(C),
  133. receive_result(C, Ref),
  134. Error;
  135. sync_on_error(_C, R) ->
  136. R.