epgsqla.erl 5.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188
  1. %%% @doc
  2. %%% Asynchronous interface.
  3. %%%
  4. %%% All the functions return `reference()' immediately. Results are delivered
  5. %%% asynchronously in a form of `{connection(), reference(), Result}', where
  6. %%% `Result' is what synchronous version of this function normally returns.
  7. %%% @end
  8. %%% Copyright (C) 2011 - Anton Lebedevich. All rights reserved.
  9. -module(epgsqla).
  10. -export([start_link/0,
  11. connect/1, connect/2, connect/3, connect/4, connect/5,
  12. close/1,
  13. get_parameter/2,
  14. set_notice_receiver/2,
  15. get_cmd_status/1,
  16. get_backend_pid/1,
  17. squery/2,
  18. equery/2, equery/3,
  19. prepared_query/3,
  20. parse/2, parse/3, parse/4,
  21. describe/2, describe/3,
  22. bind/3, bind/4,
  23. execute/2, execute/3, execute/4,
  24. execute_batch/2, execute_batch/3,
  25. close/2, close/3,
  26. sync/1,
  27. cancel/1,
  28. complete_connect/3]).
  29. -include("epgsql.hrl").
  30. %% -- client interface --
  31. -spec start_link() -> {ok, pid()}.
  32. start_link() ->
  33. epgsql_sock:start_link().
  34. connect(Opts) ->
  35. {ok, C} = epgsql_sock:start_link(),
  36. call_connect(C, Opts).
  37. connect(Host, Opts) ->
  38. connect(Host, os:getenv("USER"), "", Opts).
  39. connect(Host, Username, Opts) ->
  40. connect(Host, Username, "", Opts).
  41. connect(Host, Username, Password, Opts) ->
  42. {ok, C} = epgsql_sock:start_link(),
  43. connect(C, Host, Username, Password, Opts).
  44. -spec connect(epgsql:connection(), inet:ip_address() | inet:hostname(),
  45. string(), string(), epgsql:connect_opts()) -> reference().
  46. connect(C, Host, Username, Password, Opts) ->
  47. Opts1 = maps:merge(epgsql:to_map(Opts),
  48. #{host => Host,
  49. username => Username,
  50. password => Password}),
  51. call_connect(C, Opts1).
  52. -spec call_connect(epgsql:connection(), epgsql:connect_opts()) -> reference().
  53. call_connect(C, Opts) ->
  54. Opts1 = epgsql_cmd_connect:opts_hide_password(epgsql:to_map(Opts)),
  55. complete_connect(
  56. C, cast(C, epgsql_cmd_connect, Opts1), Opts1).
  57. -spec close(epgsql:connection()) -> ok.
  58. close(C) ->
  59. epgsql_sock:close(C).
  60. -spec get_parameter(epgsql:connection(), binary()) -> binary() | undefined.
  61. get_parameter(C, Name) ->
  62. epgsql_sock:get_parameter(C, Name).
  63. -spec set_notice_receiver(epgsql:connection(), undefined | pid() | atom()) ->
  64. {ok, Previous :: pid() | atom()}.
  65. set_notice_receiver(C, PidOrName) ->
  66. epgsql_sock:set_notice_receiver(C, PidOrName).
  67. -spec get_cmd_status(epgsql:connection()) -> {ok, Status}
  68. when
  69. Status :: undefined | atom() | {atom(), integer()}.
  70. get_cmd_status(C) ->
  71. epgsql_sock:get_cmd_status(C).
  72. -spec get_backend_pid(epgsql:connection()) -> integer().
  73. get_backend_pid(C) ->
  74. epgsql_sock:get_backend_pid(C).
  75. -spec squery(epgsql:connection(), epgsql:sql_query()) -> reference().
  76. squery(C, Sql) ->
  77. cast(C, epgsql_cmd_squery, Sql).
  78. equery(C, Sql) ->
  79. equery(C, Sql, []).
  80. -spec equery(epgsql:connection(), epgsql:statement(), [epgsql:typed_param()]) -> reference().
  81. equery(C, Statement, TypedParameters) ->
  82. cast(C, epgsql_cmd_equery, {Statement, TypedParameters}).
  83. -spec prepared_query(epgsql:connection(), epgsql:statement(), [epgsql:typed_param()]) -> reference().
  84. prepared_query(C, Statement, TypedParameters) ->
  85. cast(C, epgsql_cmd_prepared_query, {Statement, TypedParameters}).
  86. parse(C, Sql) ->
  87. parse(C, "", Sql, []).
  88. parse(C, Sql, Types) ->
  89. parse(C, "", Sql, Types).
  90. -spec parse(epgsql:connection(), iolist(), epgsql:sql_query(), [epgsql:epgsql_type()]) -> reference().
  91. parse(C, Name, Sql, Types) ->
  92. cast(C, epgsql_cmd_parse, {Name, Sql, Types}).
  93. bind(C, Statement, Parameters) ->
  94. bind(C, Statement, "", Parameters).
  95. -spec bind(epgsql:connection(), epgsql:statement(), string(), [epgsql:bind_param()]) -> reference().
  96. bind(C, Statement, PortalName, Parameters) ->
  97. cast(C, epgsql_cmd_bind, {Statement, PortalName, Parameters}).
  98. execute(C, S) ->
  99. execute(C, S, "", 0).
  100. execute(C, S, N) ->
  101. execute(C, S, "", N).
  102. -spec execute(epgsql:connection(), epgsql:statement(), string(), non_neg_integer()) -> reference().
  103. execute(C, Statement, PortalName, MaxRows) ->
  104. cast(C, epgsql_cmd_execute, {Statement, PortalName, MaxRows}).
  105. -spec execute_batch(epgsql:connection(), [{epgsql:statement(), [epgsql:bind_param()]}]) -> reference().
  106. execute_batch(C, Batch) ->
  107. cast(C, epgsql_cmd_batch, Batch).
  108. -spec execute_batch(epgsql:connection(), epgsql:statement(), [ [epgsql:bind_param()] ]) -> reference().
  109. execute_batch(C, #statement{} = Statement, Batch) ->
  110. cast(C, epgsql_cmd_batch, {Statement, Batch}).
  111. describe(C, #statement{name = Name}) ->
  112. describe(C, statement, Name).
  113. describe(C, statement, Name) ->
  114. cast(C, epgsql_cmd_describe_statement, Name);
  115. describe(C, portal, Name) ->
  116. cast(C, epgsql_cmd_describe_portal, Name).
  117. close(C, #statement{name = Name}) ->
  118. close(C, statement, Name).
  119. close(C, Type, Name) ->
  120. cast(C, epgsql_cmd_close, {Type, Name}).
  121. sync(C) ->
  122. cast(C, epgsql_cmd_sync, []).
  123. -spec cancel(epgsql:connection()) -> ok.
  124. cancel(C) ->
  125. epgsql_sock:cancel(C).
  126. %% -- internal functions --
  127. cast(C, Command, Args) ->
  128. epgsql_sock:async_command(C, cast, Command, Args).
  129. complete_connect(C, Ref, Opts) ->
  130. receive
  131. %% If we connect, then try and update the type cache. When
  132. %% all is said and done, pass the result along as a message.
  133. {C, Ref, connected} ->
  134. case maps:get(codecs, Opts, undefined) of
  135. undefined ->
  136. {ok, _} = epgsql:update_type_cache(C);
  137. [_|_] = Codecs ->
  138. {ok, _} = epgsql:update_type_cache(C, Codecs);
  139. [] ->
  140. ok
  141. end,
  142. self() ! {C, Ref, connected};
  143. {C, Ref, {error, _} = Err} ->
  144. self() ! {C, Ref, Err};
  145. {'EXIT', C, Reason} ->
  146. self() ! {'EXIT', C, Reason}
  147. end,
  148. Ref.