pgsql.erl 3.5 KB

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