epgsql_cast.erl 3.6 KB

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