epgsql_cast.erl 4.4 KB

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