aboutsummaryrefslogtreecommitdiff
path: root/src/odbc
diff options
context:
space:
mode:
Diffstat (limited to 'src/odbc')
-rw-r--r--src/odbc/Makefile.in2
-rw-r--r--src/odbc/ejabberd_odbc.erl613
-rw-r--r--src/odbc/ejabberd_odbc_sup.erl90
-rw-r--r--src/odbc/mysql.sql9
-rw-r--r--src/odbc/odbc_queries.erl1221
-rw-r--r--src/odbc/pg.sql9
6 files changed, 995 insertions, 949 deletions
diff --git a/src/odbc/Makefile.in b/src/odbc/Makefile.in
index d51439fb9..3f4898d3a 100644
--- a/src/odbc/Makefile.in
+++ b/src/odbc/Makefile.in
@@ -14,7 +14,7 @@ EFLAGS += -pz ..
# make debug=true to compile Erlang module with debug informations.
ifdef debug
- EFLAGS+=+debug_info +export_all
+ EFLAGS+=+debug_info
endif
OUTDIR = ..
diff --git a/src/odbc/ejabberd_odbc.erl b/src/odbc/ejabberd_odbc.erl
index a399a53d1..1cb157c05 100644
--- a/src/odbc/ejabberd_odbc.erl
+++ b/src/odbc/ejabberd_odbc.erl
@@ -25,6 +25,7 @@
%%%----------------------------------------------------------------------
-module(ejabberd_odbc).
+
-author('alexey@process-one.net').
-define(GEN_FSM, p1_fsm).
@@ -45,68 +46,87 @@
keep_alive/1]).
%% gen_fsm callbacks
--export([init/1,
- handle_event/3,
- handle_sync_event/4,
- handle_info/3,
- terminate/3,
- print_state/1,
+-export([init/1, handle_event/3, handle_sync_event/4,
+ handle_info/3, terminate/3, print_state/1,
code_change/4]).
%% gen_fsm states
--export([connecting/2,
- connecting/3,
- session_established/2,
- session_established/3]).
+-export([connecting/2, connecting/3,
+ session_established/2, session_established/3]).
-include("ejabberd.hrl").
--record(state, {db_ref,
- db_type,
- start_interval,
- host,
- max_pending_requests_len,
- pending_requests}).
+-record(state,
+ {db_ref = self() :: pid(),
+ db_type = odbc :: pgsql | mysql | odbc,
+ start_interval = 0 :: non_neg_integer(),
+ host = <<"">> :: binary(),
+ max_pending_requests_len :: non_neg_integer(),
+ pending_requests = {0, queue:new()} :: {non_neg_integer(), queue()}}).
-define(STATE_KEY, ejabberd_odbc_state).
+
-define(NESTING_KEY, ejabberd_odbc_nesting_level).
+
-define(TOP_LEVEL_TXN, 0).
+
-define(MAX_TRANSACTION_RESTARTS, 10).
+
-define(PGSQL_PORT, 5432).
+
-define(MYSQL_PORT, 3306).
--define(TRANSACTION_TIMEOUT, 60000). % milliseconds
+-define(TRANSACTION_TIMEOUT, 60000).
+
-define(KEEPALIVE_TIMEOUT, 60000).
--define(KEEPALIVE_QUERY, "SELECT 1;").
+
+-define(KEEPALIVE_QUERY, <<"SELECT 1;">>).
%%-define(DBGFSM, true).
-ifdef(DBGFSM).
+
-define(FSMOPTS, [{debug, [trace]}]).
+
-else.
+
-define(FSMOPTS, []).
+
-endif.
%%%----------------------------------------------------------------------
%%% API
%%%----------------------------------------------------------------------
start(Host) ->
- ?GEN_FSM:start(ejabberd_odbc, [Host], fsm_limit_opts() ++ ?FSMOPTS).
+ (?GEN_FSM):start(ejabberd_odbc, [Host],
+ fsm_limit_opts() ++ (?FSMOPTS)).
start_link(Host, StartInterval) ->
- ?GEN_FSM:start_link(ejabberd_odbc, [Host, StartInterval],
- fsm_limit_opts() ++ ?FSMOPTS).
+ (?GEN_FSM):start_link(ejabberd_odbc,
+ [Host, StartInterval],
+ fsm_limit_opts() ++ (?FSMOPTS)).
+
+-type sql_query() :: [sql_query() | binary()].
+-type sql_query_result() :: {updated, non_neg_integer()} |
+ {error, binary()} |
+ {selected, [binary()],
+ [[binary()]]}.
+
+-spec sql_query(binary(), sql_query()) -> sql_query_result().
sql_query(Host, Query) ->
sql_call(Host, {sql_query, Query}).
%% SQL transaction based on a list of queries
%% This function automatically
-sql_transaction(Host, Queries) when is_list(Queries) ->
- F = fun() ->
- lists:foreach(fun(Query) ->
- sql_query_t(Query)
- end,
+-spec sql_transaction(binary(), [sql_query()] | fun(() -> any())) ->
+ {atomic, any()} |
+ {aborted, any()}.
+
+sql_transaction(Host, Queries)
+ when is_list(Queries) ->
+ F = fun () ->
+ lists:foreach(fun (Query) -> sql_query_t(Query) end,
Queries)
end,
sql_transaction(Host, F);
@@ -115,67 +135,64 @@ sql_transaction(Host, F) when is_function(F) ->
sql_call(Host, {sql_transaction, F}).
%% SQL bloc, based on a erlang anonymous function (F = fun)
-sql_bloc(Host, F) ->
- sql_call(Host, {sql_bloc, F}).
+sql_bloc(Host, F) -> sql_call(Host, {sql_bloc, F}).
sql_call(Host, Msg) ->
case get(?STATE_KEY) of
- undefined ->
- ?GEN_FSM:sync_send_event(ejabberd_odbc_sup:get_random_pid(Host),
- {sql_cmd, Msg, now()}, ?TRANSACTION_TIMEOUT);
- _State ->
- nested_op(Msg)
+ undefined ->
+ (?GEN_FSM):sync_send_event(ejabberd_odbc_sup:get_random_pid(Host),
+ {sql_cmd, Msg, now()},
+ ?TRANSACTION_TIMEOUT);
+ _State -> nested_op(Msg)
end.
-% perform a harmless query on all opened connexions to avoid connexion close.
keep_alive(PID) ->
- ?GEN_FSM:sync_send_event(PID, {sql_cmd, {sql_query, ?KEEPALIVE_QUERY}, now()},
- ?KEEPALIVE_TIMEOUT).
+ (?GEN_FSM):sync_send_event(PID,
+ {sql_cmd, {sql_query, ?KEEPALIVE_QUERY}, now()},
+ ?KEEPALIVE_TIMEOUT).
+
+-spec sql_query_t(sql_query()) -> sql_query_result().
%% This function is intended to be used from inside an sql_transaction:
sql_query_t(Query) ->
QRes = sql_query_internal(Query),
case QRes of
- {error, Reason} ->
- throw({aborted, Reason});
- Rs when is_list(Rs) ->
- case lists:keysearch(error, 1, Rs) of
- {value, {error, Reason}} ->
- throw({aborted, Reason});
- _ ->
- QRes
- end;
- _ ->
- QRes
+ {error, Reason} -> throw({aborted, Reason});
+ Rs when is_list(Rs) ->
+ case lists:keysearch(error, 1, Rs) of
+ {value, {error, Reason}} -> throw({aborted, Reason});
+ _ -> QRes
+ end;
+ _ -> QRes
end.
%% Escape character that will confuse an SQL engine
-escape(S) when is_list(S) ->
- [odbc_queries:escape(C) || C <- S];
-escape(S) when is_binary(S) ->
- escape(binary_to_list(S)).
+escape(S) ->
+ << <<(odbc_queries:escape(Char))/binary>> || <<Char>> <= S >>.
%% Escape character that will confuse an SQL engine
%% Percent and underscore only need to be escaped for pattern matching like
%% statement
-escape_like(S) when is_list(S) ->
- [escape_like(C) || C <- S];
-escape_like($%) -> "\\%";
-escape_like($_) -> "\\_";
-escape_like(C) -> odbc_queries:escape(C).
-
-to_bool("t") -> true;
-to_bool("true") -> true;
-to_bool("1") -> true;
+escape_like(S) when is_binary(S) ->
+ << <<(escape_like(C))/binary>> || <<C>> <= S >>;
+escape_like($%) -> <<"\\%">>;
+escape_like($_) -> <<"\\_">>;
+escape_like(C) when is_integer(C), C >= 0, C =< 255 -> odbc_queries:escape(C).
+
+to_bool(<<"t">>) -> true;
+to_bool(<<"true">>) -> true;
+to_bool(<<"1">>) -> true;
to_bool(true) -> true;
to_bool(1) -> true;
to_bool(_) -> false.
encode_term(Term) ->
- escape(lists:flatten(io_lib:print(Term))).
+ escape(list_to_binary(
+ erl_prettypr:format(erl_syntax:abstract(Term)))).
-decode_term(Str) ->
- {ok, Tokens, _} = erl_scan:string(Str ++ "."),
+decode_term(Bin) ->
+ Str = binary_to_list(<<Bin/binary, ".">>),
+ {ok, Tokens, _} = erl_scan:string(Str),
{ok, Term} = erl_parse:parse_term(Tokens),
Term.
@@ -183,76 +200,83 @@ decode_term(Str) ->
%%% Callback functions from gen_fsm
%%%----------------------------------------------------------------------
init([Host, StartInterval]) ->
- case ejabberd_config:get_local_option({odbc_keepalive_interval, Host}) of
- KeepaliveInterval when is_integer(KeepaliveInterval) ->
- timer:apply_interval(KeepaliveInterval*1000, ?MODULE,
- keep_alive, [self()]);
- undefined ->
- ok;
- _Other ->
- ?ERROR_MSG("Wrong odbc_keepalive_interval definition '~p'"
- " for host ~p.~n", [_Other, Host])
+ case ejabberd_config:get_local_option(
+ {odbc_keepalive_interval, Host},
+ fun(I) when is_integer(I), I>0 -> I end) of
+ undefined ->
+ ok;
+ KeepaliveInterval ->
+ timer:apply_interval(KeepaliveInterval * 1000, ?MODULE,
+ keep_alive, [self()])
end,
[DBType | _] = db_opts(Host),
- ?GEN_FSM:send_event(self(), connect),
+ (?GEN_FSM):send_event(self(), connect),
ejabberd_odbc_sup:add_pid(Host, self()),
- {ok, connecting, #state{db_type = DBType,
- host = Host,
- max_pending_requests_len = max_fsm_queue(),
- pending_requests = {0, queue:new()},
- start_interval = StartInterval}}.
+ {ok, connecting,
+ #state{db_type = DBType, host = Host,
+ max_pending_requests_len = max_fsm_queue(),
+ pending_requests = {0, queue:new()},
+ start_interval = StartInterval}}.
connecting(connect, #state{host = Host} = State) ->
ConnectRes = case db_opts(Host) of
- [mysql | Args] ->
- apply(fun mysql_connect/5, Args);
- [pgsql | Args] ->
- apply(fun pgsql_connect/5, Args);
- [odbc | Args] ->
- apply(fun odbc_connect/1, Args)
+ [mysql | Args] -> apply(fun mysql_connect/5, Args);
+ [pgsql | Args] -> apply(fun pgsql_connect/5, Args);
+ [odbc | Args] -> apply(fun odbc_connect/1, Args)
end,
{_, PendingRequests} = State#state.pending_requests,
case ConnectRes of
- {ok, Ref} ->
- erlang:monitor(process, Ref),
- lists:foreach(
- fun(Req) ->
- ?GEN_FSM:send_event(self(), Req)
- end, queue:to_list(PendingRequests)),
- {next_state, session_established,
- State#state{db_ref = Ref,
- pending_requests = {0, queue:new()}}};
- {error, Reason} ->
- ?INFO_MSG("~p connection failed:~n"
- "** Reason: ~p~n"
- "** Retry after: ~p seconds",
- [State#state.db_type, Reason,
- State#state.start_interval div 1000]),
- ?GEN_FSM:send_event_after(State#state.start_interval,
+ {ok, Ref} ->
+ erlang:monitor(process, Ref),
+ lists:foreach(fun (Req) ->
+ (?GEN_FSM):send_event(self(), Req)
+ end,
+ queue:to_list(PendingRequests)),
+ {next_state, session_established,
+ State#state{db_ref = Ref,
+ pending_requests = {0, queue:new()}}};
+ {error, Reason} ->
+ ?INFO_MSG("~p connection failed:~n** Reason: ~p~n** "
+ "Retry after: ~p seconds",
+ [State#state.db_type, Reason,
+ State#state.start_interval div 1000]),
+ (?GEN_FSM):send_event_after(State#state.start_interval,
connect),
- {next_state, connecting, State}
+ {next_state, connecting, State}
end;
connecting(Event, State) ->
- ?WARNING_MSG("unexpected event in 'connecting': ~p", [Event]),
+ ?WARNING_MSG("unexpected event in 'connecting': ~p",
+ [Event]),
{next_state, connecting, State}.
-connecting({sql_cmd, {sql_query, ?KEEPALIVE_QUERY}, _Timestamp}, From, State) ->
- ?GEN_FSM:reply(From, {error, "SQL connection failed"}),
+connecting({sql_cmd, {sql_query, ?KEEPALIVE_QUERY},
+ _Timestamp},
+ From, State) ->
+ (?GEN_FSM):reply(From,
+ {error, <<"SQL connection failed">>}),
{next_state, connecting, State};
-connecting({sql_cmd, Command, Timestamp} = Req, From, State) ->
- ?DEBUG("queuing pending request while connecting:~n\t~p", [Req]),
+connecting({sql_cmd, Command, Timestamp} = Req, From,
+ State) ->
+ ?DEBUG("queuing pending request while connecting:~n\t~p",
+ [Req]),
{Len, PendingRequests} = State#state.pending_requests,
- NewPendingRequests =
- if Len < State#state.max_pending_requests_len ->
- {Len + 1, queue:in({sql_cmd, Command, From, Timestamp}, PendingRequests)};
- true ->
- lists:foreach(
- fun({sql_cmd, _, To, _Timestamp}) ->
- ?GEN_FSM:reply(
- To, {error, "SQL connection failed"})
- end, queue:to_list(PendingRequests)),
- {1, queue:from_list([{sql_cmd, Command, From, Timestamp}])}
- end,
+ NewPendingRequests = if Len <
+ State#state.max_pending_requests_len ->
+ {Len + 1,
+ queue:in({sql_cmd, Command, From, Timestamp},
+ PendingRequests)};
+ true ->
+ lists:foreach(fun ({sql_cmd, _, To,
+ _Timestamp}) ->
+ (?GEN_FSM):reply(To,
+ {error,
+ <<"SQL connection failed">>})
+ end,
+ queue:to_list(PendingRequests)),
+ {1,
+ queue:from_list([{sql_cmd, Command, From,
+ Timestamp}])}
+ end,
{next_state, connecting,
State#state{pending_requests = NewPendingRequests}};
connecting(Request, {Who, _Ref}, State) ->
@@ -260,17 +284,21 @@ connecting(Request, {Who, _Ref}, State) ->
[Request, Who]),
{reply, {error, badarg}, connecting, State}.
-session_established({sql_cmd, Command, Timestamp}, From, State) ->
+session_established({sql_cmd, Command, Timestamp}, From,
+ State) ->
run_sql_cmd(Command, From, State, Timestamp);
session_established(Request, {Who, _Ref}, State) ->
- ?WARNING_MSG("unexpected call ~p from ~p in 'session_established'",
+ ?WARNING_MSG("unexpected call ~p from ~p in 'session_establ"
+ "ished'",
[Request, Who]),
{reply, {error, badarg}, session_established, State}.
-session_established({sql_cmd, Command, From, Timestamp}, State) ->
+session_established({sql_cmd, Command, From, Timestamp},
+ State) ->
run_sql_cmd(Command, From, State, Timestamp);
session_established(Event, State) ->
- ?WARNING_MSG("unexpected event in 'session_established': ~p", [Event]),
+ ?WARNING_MSG("unexpected event in 'session_established': ~p",
+ [Event]),
{next_state, session_established, State}.
handle_event(_Event, StateName, State) ->
@@ -284,22 +312,20 @@ code_change(_OldVsn, StateName, State, _Extra) ->
%% We receive the down signal when we loose the MySQL connection (we are
%% monitoring the connection)
-handle_info({'DOWN', _MonitorRef, process, _Pid, _Info}, _StateName, State) ->
- ?GEN_FSM:send_event(self(), connect),
+handle_info({'DOWN', _MonitorRef, process, _Pid, _Info},
+ _StateName, State) ->
+ (?GEN_FSM):send_event(self(), connect),
{next_state, connecting, State};
handle_info(Info, StateName, State) ->
- ?WARNING_MSG("unexpected info in ~p: ~p", [StateName, Info]),
+ ?WARNING_MSG("unexpected info in ~p: ~p",
+ [StateName, Info]),
{next_state, StateName, State}.
terminate(_Reason, _StateName, State) ->
ejabberd_odbc_sup:remove_pid(State#state.host, self()),
case State#state.db_type of
- mysql ->
- %% old versions of mysql driver don't have the stop function
- %% so the catch
- catch mysql_conn:stop(State#state.db_ref);
- _ ->
- ok
+ mysql -> catch mysql_conn:stop(State#state.db_ref);
+ _ -> ok
end,
ok.
@@ -308,23 +334,23 @@ terminate(_Reason, _StateName, State) ->
%% Purpose: Prepare the state to be printed on error log
%% Returns: State to print
%%----------------------------------------------------------------------
-print_state(State) ->
- State.
+print_state(State) -> State.
+
%%%----------------------------------------------------------------------
%%% Internal functions
%%%----------------------------------------------------------------------
run_sql_cmd(Command, From, State, Timestamp) ->
case timer:now_diff(now(), Timestamp) div 1000 of
- Age when Age < ?TRANSACTION_TIMEOUT ->
- put(?NESTING_KEY, ?TOP_LEVEL_TXN),
- put(?STATE_KEY, State),
- abort_on_driver_error(outer_op(Command), From);
- Age ->
- ?ERROR_MSG("Database was not available or too slow,"
- " discarding ~p milliseconds old request~n~p~n",
- [Age, Command]),
- {next_state, session_established, State}
+ Age when Age < (?TRANSACTION_TIMEOUT) ->
+ put(?NESTING_KEY, ?TOP_LEVEL_TXN),
+ put(?STATE_KEY, State),
+ abort_on_driver_error(outer_op(Command), From);
+ Age ->
+ ?ERROR_MSG("Database was not available or too slow, "
+ "discarding ~p milliseconds old request~n~p~n",
+ [Age, Command]),
+ {next_state, session_established, State}
end.
%% Only called by handle_call, only handles top level operations.
@@ -332,143 +358,125 @@ run_sql_cmd(Command, From, State, Timestamp) ->
outer_op({sql_query, Query}) ->
sql_query_internal(Query);
outer_op({sql_transaction, F}) ->
- outer_transaction(F, ?MAX_TRANSACTION_RESTARTS, "");
-outer_op({sql_bloc, F}) ->
- execute_bloc(F).
+ outer_transaction(F, ?MAX_TRANSACTION_RESTARTS, <<"">>);
+outer_op({sql_bloc, F}) -> execute_bloc(F).
%% Called via sql_query/transaction/bloc from client code when inside a
%% nested operation
nested_op({sql_query, Query}) ->
- %% XXX - use sql_query_t here insted? Most likely would break
- %% callers who expect {error, _} tuples (sql_query_t turns
- %% these into throws)
sql_query_internal(Query);
nested_op({sql_transaction, F}) ->
NestingLevel = get(?NESTING_KEY),
- if NestingLevel =:= ?TOP_LEVEL_TXN ->
- %% First transaction inside a (series of) sql_blocs
- outer_transaction(F, ?MAX_TRANSACTION_RESTARTS, "");
- true ->
- %% Transaction inside a transaction
- inner_transaction(F)
+ if NestingLevel =:= (?TOP_LEVEL_TXN) ->
+ outer_transaction(F, ?MAX_TRANSACTION_RESTARTS, <<"">>);
+ true -> inner_transaction(F)
end;
-nested_op({sql_bloc, F}) ->
- execute_bloc(F).
+nested_op({sql_bloc, F}) -> execute_bloc(F).
%% Never retry nested transactions - only outer transactions
inner_transaction(F) ->
PreviousNestingLevel = get(?NESTING_KEY),
case get(?NESTING_KEY) of
- ?TOP_LEVEL_TXN ->
- {backtrace, T} = process_info(self(), backtrace),
- ?ERROR_MSG("inner transaction called at outer txn level. Trace: ~s",
- [T]),
- erlang:exit(implementation_faulty);
- _N -> ok
+ ?TOP_LEVEL_TXN ->
+ {backtrace, T} = process_info(self(), backtrace),
+ ?ERROR_MSG("inner transaction called at outer txn "
+ "level. Trace: ~s",
+ [T]),
+ erlang:exit(implementation_faulty);
+ _N -> ok
end,
put(?NESTING_KEY, PreviousNestingLevel + 1),
Result = (catch F()),
put(?NESTING_KEY, PreviousNestingLevel),
case Result of
- {aborted, Reason} ->
- {aborted, Reason};
- {'EXIT', Reason} ->
- {'EXIT', Reason};
- {atomic, Res} ->
- {atomic, Res};
- Res ->
- {atomic, Res}
+ {aborted, Reason} -> {aborted, Reason};
+ {'EXIT', Reason} -> {'EXIT', Reason};
+ {atomic, Res} -> {atomic, Res};
+ Res -> {atomic, Res}
end.
outer_transaction(F, NRestarts, _Reason) ->
PreviousNestingLevel = get(?NESTING_KEY),
case get(?NESTING_KEY) of
- ?TOP_LEVEL_TXN ->
- ok;
- _N ->
- {backtrace, T} = process_info(self(), backtrace),
- ?ERROR_MSG("outer transaction called at inner txn level. Trace: ~s",
- [T]),
- erlang:exit(implementation_faulty)
+ ?TOP_LEVEL_TXN -> ok;
+ _N ->
+ {backtrace, T} = process_info(self(), backtrace),
+ ?ERROR_MSG("outer transaction called at inner txn "
+ "level. Trace: ~s",
+ [T]),
+ erlang:exit(implementation_faulty)
end,
- sql_query_internal("begin;"),
+ sql_query_internal(<<"begin;">>),
put(?NESTING_KEY, PreviousNestingLevel + 1),
Result = (catch F()),
put(?NESTING_KEY, PreviousNestingLevel),
case Result of
- {aborted, Reason} when NRestarts > 0 ->
- %% Retry outer transaction upto NRestarts times.
- sql_query_internal("rollback;"),
- outer_transaction(F, NRestarts - 1, Reason);
- {aborted, Reason} when NRestarts =:= 0 ->
- %% Too many retries of outer transaction.
- ?ERROR_MSG("SQL transaction restarts exceeded~n"
- "** Restarts: ~p~n"
- "** Last abort reason: ~p~n"
- "** Stacktrace: ~p~n"
- "** When State == ~p",
- [?MAX_TRANSACTION_RESTARTS, Reason,
- erlang:get_stacktrace(), get(?STATE_KEY)]),
- sql_query_internal("rollback;"),
- {aborted, Reason};
- {'EXIT', Reason} ->
- %% Abort sql transaction on EXIT from outer txn only.
- sql_query_internal("rollback;"),
- {aborted, Reason};
- Res ->
- %% Commit successful outer txn
- sql_query_internal("commit;"),
- {atomic, Res}
+ {aborted, Reason} when NRestarts > 0 ->
+ sql_query_internal(<<"rollback;">>),
+ outer_transaction(F, NRestarts - 1, Reason);
+ {aborted, Reason} when NRestarts =:= 0 ->
+ ?ERROR_MSG("SQL transaction restarts exceeded~n** "
+ "Restarts: ~p~n** Last abort reason: "
+ "~p~n** Stacktrace: ~p~n** When State "
+ "== ~p",
+ [?MAX_TRANSACTION_RESTARTS, Reason,
+ erlang:get_stacktrace(), get(?STATE_KEY)]),
+ sql_query_internal(<<"rollback;">>),
+ {aborted, Reason};
+ {'EXIT', Reason} ->
+ sql_query_internal(<<"rollback;">>), {aborted, Reason};
+ Res -> sql_query_internal(<<"commit;">>), {atomic, Res}
end.
execute_bloc(F) ->
- %% We don't alter ?NESTING_KEY here as only SQL transactions alter
- %% txn nesting
case catch F() of
- {aborted, Reason} ->
- {aborted, Reason};
- {'EXIT', Reason} ->
- {aborted, Reason};
- Res ->
- {atomic, Res}
+ {aborted, Reason} -> {aborted, Reason};
+ {'EXIT', Reason} -> {aborted, Reason};
+ Res -> {atomic, Res}
end.
sql_query_internal(Query) ->
State = get(?STATE_KEY),
Res = case State#state.db_type of
- odbc ->
- odbc:sql_query(State#state.db_ref, Query);
- pgsql ->
- pgsql_to_odbc(pgsql:squery(State#state.db_ref, Query));
- mysql ->
- ?DEBUG("MySQL, Send query~n~p~n", [Query]),
- R = mysql_to_odbc(mysql_conn:fetch(State#state.db_ref,
- Query, self())),
- %% ?INFO_MSG("MySQL, Received result~n~p~n", [R]),
- R
- end,
+ odbc ->
+ to_odbc(odbc:sql_query(State#state.db_ref, Query,
+ (?TRANSACTION_TIMEOUT) - 1000));
+ pgsql ->
+ pgsql_to_odbc(pgsql:squery(State#state.db_ref, Query));
+ mysql ->
+ ?DEBUG("MySQL, Send query~n~p~n", [Query]),
+ %%squery to be able to specify result_type = binary
+ %%[Query] because mysql_conn expect query to be a list (elements can be binaries, or iolist)
+ %% but doesn't accept just a binary
+ R = mysql_to_odbc(mysql_conn:squery(State#state.db_ref,
+ [Query], self(),
+ [{timeout, (?TRANSACTION_TIMEOUT) - 1000},
+ {result_type, binary}])),
+ %% ?INFO_MSG("MySQL, Received result~n~p~n", [R]),
+ R
+ end,
case Res of
- {error, "No SQL-driver information available."} ->
- % workaround for odbc bug
- {updated, 0};
- _Else -> Res
+ {error, <<"No SQL-driver information available.">>} ->
+ {updated, 0};
+ _Else -> Res
end.
%% Generate the OTP callback return tuple depending on the driver result.
-abort_on_driver_error({error, "query timed out"} = Reply, From) ->
- %% mysql driver error
- ?GEN_FSM:reply(From, Reply),
+abort_on_driver_error({error, <<"query timed out">>} =
+ Reply,
+ From) ->
+ (?GEN_FSM):reply(From, Reply),
{stop, timeout, get(?STATE_KEY)};
-abort_on_driver_error({error, "Failed sending data on socket" ++ _} = Reply,
+abort_on_driver_error({error,
+ <<"Failed sending data on socket", _/binary>>} =
+ Reply,
From) ->
- %% mysql driver error
- ?GEN_FSM:reply(From, Reply),
+ (?GEN_FSM):reply(From, Reply),
{stop, closed, get(?STATE_KEY)};
abort_on_driver_error(Reply, From) ->
- ?GEN_FSM:reply(From, Reply),
+ (?GEN_FSM):reply(From, Reply),
{next_state, session_established, get(?STATE_KEY)}.
-
%% == pure ODBC code
%% part of init/1
@@ -482,44 +490,54 @@ odbc_connect(SQLServer) ->
%% part of init/1
%% Open a database connection to PostgreSQL
pgsql_connect(Server, Port, DB, Username, Password) ->
- pgsql:connect(Server, DB, Username, Password, Port).
+ case pgsql:connect([{host, Server},
+ {database, DB},
+ {user, Username},
+ {password, Password},
+ {port, Port},
+ {as_binary, true}]) of
+ {ok, Ref} ->
+ pgsql:squery(Ref, [<<"alter database ">>, DB, <<" set ">>,
+ <<"standard_conforming_strings='off';">>]),
+ {ok, Ref};
+ Err ->
+ Err
+ end.
%% Convert PostgreSQL query result to Erlang ODBC result formalism
pgsql_to_odbc({ok, PGSQLResult}) ->
case PGSQLResult of
- [Item] ->
- pgsql_item_to_odbc(Item);
- Items ->
- [pgsql_item_to_odbc(Item) || Item <- Items]
+ [Item] -> pgsql_item_to_odbc(Item);
+ Items -> [pgsql_item_to_odbc(Item) || Item <- Items]
end.
-pgsql_item_to_odbc({"SELECT" ++ _, Rows, Recs}) ->
- {selected,
- [element(1, Row) || Row <- Rows],
- [list_to_tuple(Rec) || Rec <- Recs]};
-pgsql_item_to_odbc("INSERT " ++ OIDN) ->
- [_OID, N] = string:tokens(OIDN, " "),
- {updated, list_to_integer(N)};
-pgsql_item_to_odbc("DELETE " ++ N) ->
- {updated, list_to_integer(N)};
-pgsql_item_to_odbc("UPDATE " ++ N) ->
- {updated, list_to_integer(N)};
-pgsql_item_to_odbc({error, Error}) ->
- {error, Error};
-pgsql_item_to_odbc(_) ->
- {updated,undefined}.
+pgsql_item_to_odbc({<<"SELECT", _/binary>>, Rows,
+ Recs}) ->
+ {selected, [element(1, Row) || Row <- Rows], Recs};
+pgsql_item_to_odbc(<<"INSERT ", OIDN/binary>>) ->
+ [_OID, N] = str:tokens(OIDN, <<" ">>),
+ {updated, jlib:binary_to_integer(N)};
+pgsql_item_to_odbc(<<"DELETE ", N/binary>>) ->
+ {updated, jlib:binary_to_integer(N)};
+pgsql_item_to_odbc(<<"UPDATE ", N/binary>>) ->
+ {updated, jlib:binary_to_integer(N)};
+pgsql_item_to_odbc({error, Error}) -> {error, Error};
+pgsql_item_to_odbc(_) -> {updated, undefined}.
%% == Native MySQL code
%% part of init/1
%% Open a database connection to MySQL
mysql_connect(Server, Port, DB, Username, Password) ->
- case mysql_conn:start(Server, Port, Username, Password, DB, fun log/3) of
- {ok, Ref} ->
- mysql_conn:fetch(Ref, ["set names 'utf8';"], self()),
- {ok, Ref};
- Err ->
- Err
+ case mysql_conn:start(binary_to_list(Server), Port,
+ binary_to_list(Username), binary_to_list(Password),
+ binary_to_list(DB), fun log/3)
+ of
+ {ok, Ref} ->
+ mysql_conn:fetch(Ref, [<<"set names 'utf8';">>],
+ self()),
+ {ok, Ref};
+ Err -> Err
end.
%% Convert MySQL query result to Erlang ODBC result formalism
@@ -528,58 +546,67 @@ mysql_to_odbc({updated, MySQLRes}) ->
mysql_to_odbc({data, MySQLRes}) ->
mysql_item_to_odbc(mysql:get_result_field_info(MySQLRes),
mysql:get_result_rows(MySQLRes));
-mysql_to_odbc({error, MySQLRes}) when is_list(MySQLRes) ->
+mysql_to_odbc({error, MySQLRes})
+ when is_binary(MySQLRes) ->
{error, MySQLRes};
mysql_to_odbc({error, MySQLRes}) ->
{error, mysql:get_result_reason(MySQLRes)}.
%% When tabular data is returned, convert it to the ODBC formalism
mysql_item_to_odbc(Columns, Recs) ->
- %% For now, there is a bug and we do not get the correct value from MySQL
- %% module:
- {selected,
- [element(2, Column) || Column <- Columns],
- [list_to_tuple(Rec) || Rec <- Recs]}.
+ {selected, [element(2, Column) || Column <- Columns], Recs}.
+
+to_odbc({selected, Columns, Recs}) ->
+ {selected, Columns, [tuple_to_list(Rec) || Rec <- Recs]};
+to_odbc(Res) ->
+ Res.
-% log function used by MySQL driver
log(Level, Format, Args) ->
case Level of
- debug ->
- ?DEBUG(Format, Args);
- normal ->
- ?INFO_MSG(Format, Args);
- error ->
- ?ERROR_MSG(Format, Args)
+ debug -> ?DEBUG(Format, Args);
+ normal -> ?INFO_MSG(Format, Args);
+ error -> ?ERROR_MSG(Format, Args)
end.
db_opts(Host) ->
- case ejabberd_config:get_local_option({odbc_server, Host}) of
- %% Default pgsql port
- {pgsql, Server, DB, User, Pass} ->
- [pgsql, Server, ?PGSQL_PORT, DB, User, Pass];
- {pgsql, Server, Port, DB, User, Pass} when is_integer(Port) ->
- [pgsql, Server, Port, DB, User, Pass];
- %% Default mysql port
- {mysql, Server, DB, User, Pass} ->
- [mysql, Server, ?MYSQL_PORT, DB, User, Pass];
- {mysql, Server, Port, DB, User, Pass} when is_integer(Port) ->
- [mysql, Server, Port, DB, User, Pass];
- SQLServer when is_list(SQLServer) ->
- [odbc, SQLServer]
+ case ejabberd_config:get_local_option(
+ {odbc_server, Host},
+ fun({Type, Server, DB, User, Pass}) ->
+ {Type,
+ iolist_to_binary(Server),
+ case Type of
+ mysql -> ?MYSQL_PORT;
+ pgsql -> ?PGSQL_PORT
+ end,
+ iolist_to_binary(DB),
+ iolist_to_binary(User),
+ iolist_to_binary(Pass)};
+ ({Type, Server, Port, DB, User, Pass})
+ when ((Type == mysql) or (Type == pgsql))
+ and (is_integer(Port) and ((Port > 0)
+ and (Port < 65536))) ->
+ {Type,
+ iolist_to_binary(Server),
+ Port,
+ iolist_to_binary(DB),
+ iolist_to_binary(User),
+ iolist_to_binary(Pass)};
+ (S) ->
+ iolist_to_binary(S)
+ end, <<"localhost">>) of
+ {Type, Server, Port, DB, User, Pass} ->
+ [Type, Server, Port, DB, User, Pass];
+ SQLServer ->
+ [odbc, SQLServer]
end.
max_fsm_queue() ->
- case ejabberd_config:get_local_option(max_fsm_queue) of
- N when is_integer(N), N>0 ->
- N;
- _ ->
- undefined
- end.
+ ejabberd_config:get_local_option(
+ max_fsm_queue,
+ fun(N) when is_integer(N), N > 0 -> N end).
fsm_limit_opts() ->
case max_fsm_queue() of
- N when is_integer(N) ->
- [{max_queue, N}];
- _ ->
- []
+ N when is_integer(N) -> [{max_queue, N}];
+ _ -> []
end.
diff --git a/src/odbc/ejabberd_odbc_sup.erl b/src/odbc/ejabberd_odbc_sup.erl
index 442b0f906..0c748d147 100644
--- a/src/odbc/ejabberd_odbc_sup.erl
+++ b/src/odbc/ejabberd_odbc_sup.erl
@@ -25,79 +25,53 @@
%%%----------------------------------------------------------------------
-module(ejabberd_odbc_sup).
+
-author('alexey@process-one.net').
%% API
--export([start_link/1,
- init/1,
- add_pid/2,
- remove_pid/2,
- get_pids/1,
- get_random_pid/1
- ]).
+-export([start_link/1, init/1, add_pid/2, remove_pid/2,
+ get_pids/1, get_random_pid/1]).
-include("ejabberd.hrl").
-define(DEFAULT_POOL_SIZE, 10).
--define(DEFAULT_ODBC_START_INTERVAL, 30). % 30 seconds
-% time to wait for the supervisor to start its child before returning
-% a timeout error to the request
--define(CONNECT_TIMEOUT, 500). % milliseconds
+-define(DEFAULT_ODBC_START_INTERVAL, 30).
+-define(CONNECT_TIMEOUT, 500).
-record(sql_pool, {host, pid}).
start_link(Host) ->
mnesia:create_table(sql_pool,
- [{ram_copies, [node()]},
- {type, bag},
+ [{ram_copies, [node()]}, {type, bag},
{local_content, true},
{attributes, record_info(fields, sql_pool)}]),
mnesia:add_table_copy(sql_pool, node(), ram_copies),
- F = fun() ->
- mnesia:delete({sql_pool, Host})
- end,
+ F = fun () -> mnesia:delete({sql_pool, Host}) end,
mnesia:ets(F),
- supervisor:start_link({local, gen_mod:get_module_proc(Host, ?MODULE)},
+ supervisor:start_link({local,
+ gen_mod:get_module_proc(Host, ?MODULE)},
?MODULE, [Host]).
init([Host]) ->
- PoolSize = case ejabberd_config:get_local_option({odbc_pool_size, Host}) of
- I when is_integer(I) ->
- I;
- undefined ->
- ?DEFAULT_POOL_SIZE;
- Other ->
- ?ERROR_MSG("Wrong odbc_pool_size definition '~p' "
- "for host ~p, default to ~p~n",
- [Other, Host, ?DEFAULT_POOL_SIZE]),
- ?DEFAULT_POOL_SIZE
- end,
- StartInterval = case ejabberd_config:get_local_option({odbc_start_interval,
- Host}) of
- Interval when is_integer(Interval) ->
- Interval;
- undefined ->
- ?DEFAULT_ODBC_START_INTERVAL;
- _Other2 ->
- ?ERROR_MSG("Wrong odbc_start_interval "
- "definition '~p' for host ~p, "
- "defaulting to ~p~n",
- [_Other2, Host,
- ?DEFAULT_ODBC_START_INTERVAL]),
- ?DEFAULT_ODBC_START_INTERVAL
- end,
- {ok, {{one_for_one, PoolSize*10, 1},
- lists:map(
- fun(I) ->
- {I,
- {ejabberd_odbc, start_link, [Host, StartInterval*1000]},
- transient,
- 2000,
- worker,
- [?MODULE]}
- end, lists:seq(1, PoolSize))}}.
+ PoolSize = ejabberd_config:get_local_option(
+ {odbc_pool_size, Host},
+ fun(I) when is_integer(I), I>0 -> I end,
+ ?DEFAULT_POOL_SIZE),
+ StartInterval = ejabberd_config:get_local_option(
+ {odbc_start_interval, Host},
+ fun(I) when is_integer(I), I>0 -> I end,
+ ?DEFAULT_ODBC_START_INTERVAL),
+ {ok,
+ {{one_for_one, PoolSize * 10, 1},
+ lists:map(fun (I) ->
+ {I,
+ {ejabberd_odbc, start_link,
+ [Host, StartInterval * 1000]},
+ transient, 2000, worker, [?MODULE]}
+ end,
+ lists:seq(1, PoolSize))}}.
get_pids(Host) ->
Rs = mnesia:dirty_read(sql_pool, Host),
@@ -108,17 +82,13 @@ get_random_pid(Host) ->
lists:nth(erlang:phash(now(), length(Pids)), Pids).
add_pid(Host, Pid) ->
- F = fun() ->
- mnesia:write(
- #sql_pool{host = Host,
- pid = Pid})
+ F = fun () ->
+ mnesia:write(#sql_pool{host = Host, pid = Pid})
end,
mnesia:ets(F).
remove_pid(Host, Pid) ->
- F = fun() ->
- mnesia:delete_object(
- #sql_pool{host = Host,
- pid = Pid})
+ F = fun () ->
+ mnesia:delete_object(#sql_pool{host = Host, pid = Pid})
end,
mnesia:ets(F).
diff --git a/src/odbc/mysql.sql b/src/odbc/mysql.sql
index 17c7d9b76..976230117 100644
--- a/src/odbc/mysql.sql
+++ b/src/odbc/mysql.sql
@@ -273,3 +273,12 @@ CREATE TABLE motd (
xml text,
created_at timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP
) CHARACTER SET utf8;
+
+CREATE TABLE caps_features (
+ node varchar(250) NOT NULL,
+ subnode varchar(250) NOT NULL,
+ feature text,
+ created_at timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP
+) CHARACTER SET utf8;
+
+CREATE INDEX i_caps_features_node_subnode ON caps_features(node(75), subnode(75));
diff --git a/src/odbc/odbc_queries.erl b/src/odbc/odbc_queries.erl
index bd4812609..66da7906f 100644
--- a/src/odbc/odbc_queries.erl
+++ b/src/odbc/odbc_queries.erl
@@ -25,52 +25,28 @@
%%%----------------------------------------------------------------------
-module(odbc_queries).
+
-author("mremond@process-one.net").
--export([get_db_type/0,
- update_t/4,
- sql_transaction/2,
- get_last/2,
- set_last_t/4,
- del_last/2,
- get_password/2,
- set_password_t/3,
- add_user/3,
- del_user/2,
- del_user_return_password/3,
- list_users/1,
- list_users/2,
- users_number/1,
- users_number/2,
- add_spool_sql/2,
- add_spool/2,
- get_and_del_spool_msg_t/2,
- del_spool_msg/2,
- get_roster/2,
- get_roster_jid_groups/2,
- get_roster_groups/3,
- del_user_roster_t/2,
- get_roster_by_jid/3,
- get_rostergroup_by_jid/3,
- del_roster/3,
- del_roster_sql/2,
- update_roster/5,
- update_roster_sql/4,
- roster_subscribe/4,
- get_subscription/3,
- set_private_data/4,
- set_private_data_sql/3,
- get_private_data/3,
- del_user_private_storage/2,
- get_default_privacy_list/2,
- get_default_privacy_list_t/1,
- get_privacy_list_names/2,
- get_privacy_list_names_t/1,
- get_privacy_list_id/3,
- get_privacy_list_id_t/2,
- get_privacy_list_data/3,
+-export([get_db_type/0, update_t/4, sql_transaction/2,
+ get_last/2, set_last_t/4, del_last/2, get_password/2,
+ set_password_t/3, add_user/3, del_user/2,
+ del_user_return_password/3, list_users/1, list_users/2,
+ users_number/1, users_number/2, add_spool_sql/2,
+ add_spool/2, get_and_del_spool_msg_t/2, del_spool_msg/2,
+ get_roster/2, get_roster_jid_groups/2,
+ get_roster_groups/3, del_user_roster_t/2,
+ get_roster_by_jid/3, get_rostergroup_by_jid/3,
+ del_roster/3, del_roster_sql/2, update_roster/5,
+ update_roster_sql/4, roster_subscribe/4,
+ get_subscription/3, set_private_data/4,
+ set_private_data_sql/3, get_private_data/3, get_private_data/2,
+ del_user_private_storage/2, get_default_privacy_list/2,
+ get_default_privacy_list_t/1, get_privacy_list_names/2,
+ get_privacy_list_names_t/1, get_privacy_list_id/3,
+ get_privacy_list_id_t/2, get_privacy_list_data/3,
get_privacy_list_data_by_id/2,
- get_privacy_list_data_by_id_t/1,
+ get_privacy_list_data_by_id_t/1,
set_default_privacy_list/2,
unset_default_privacy_list/2,
remove_privacy_list/2,
@@ -88,8 +64,11 @@
%-define(generic, true).
%-define(mssql, true).
-ifndef(mssql).
+
-undef(generic).
+
-define(generic, true).
+
-endif.
-include("ejabberd.hrl").
@@ -97,49 +76,49 @@
%% Almost a copy of string:join/2.
%% We use this version because string:join/2 is relatively
%% new function (introduced in R12B-0).
-join([], _Sep) ->
- [];
-join([H|T], Sep) ->
- [H, [[Sep, X] || X <- T]].
+join([], _Sep) -> [];
+join([H | T], Sep) -> [H, [[Sep, X] || X <- T]].
%% -----------------
%% Generic queries
-ifdef(generic).
-get_db_type() ->
- generic.
+get_db_type() -> generic.
%% Safe atomic update.
update_t(Table, Fields, Vals, Where) ->
- UPairs = lists:zipwith(fun(A, B) -> A ++ "='" ++ B ++ "'" end,
+ UPairs = lists:zipwith(fun (A, B) ->
+ <<A/binary, "='", B/binary, "'">>
+ end,
Fields, Vals),
- case ejabberd_odbc:sql_query_t(
- ["update ", Table, " set ",
- join(UPairs, ", "),
- " where ", Where, ";"]) of
- {updated, 1} ->
- ok;
- _ ->
- ejabberd_odbc:sql_query_t(
- ["insert into ", Table, "(", join(Fields, ", "),
- ") values ('", join(Vals, "', '"), "');"])
+ case ejabberd_odbc:sql_query_t([<<"update ">>, Table,
+ <<" set ">>, join(UPairs, <<", ">>),
+ <<" where ">>, Where, <<";">>])
+ of
+ {updated, 1} -> ok;
+ _ ->
+ ejabberd_odbc:sql_query_t([<<"insert into ">>, Table,
+ <<"(">>, join(Fields, <<", ">>),
+ <<") values ('">>, join(Vals, <<"', '">>),
+ <<"');">>])
end.
update(LServer, Table, Fields, Vals, Where) ->
- UPairs = lists:zipwith(fun(A, B) -> A ++ "='" ++ B ++ "'" end,
+ UPairs = lists:zipwith(fun (A, B) ->
+ <<A/binary, "='", B/binary, "'">>
+ end,
Fields, Vals),
- case ejabberd_odbc:sql_query(
- LServer,
- ["update ", Table, " set ",
- join(UPairs, ", "),
- " where ", Where, ";"]) of
- {updated, 1} ->
- ok;
- _ ->
- ejabberd_odbc:sql_query(
- LServer,
- ["insert into ", Table, "(", join(Fields, ", "),
- ") values ('", join(Vals, "', '"), "');"])
+ case ejabberd_odbc:sql_query(LServer,
+ [<<"update ">>, Table, <<" set ">>,
+ join(UPairs, <<", ">>), <<" where ">>, Where,
+ <<";">>])
+ of
+ {updated, 1} -> ok;
+ _ ->
+ ejabberd_odbc:sql_query(LServer,
+ [<<"insert into ">>, Table, <<"(">>,
+ join(Fields, <<", ">>), <<") values ('">>,
+ join(Vals, <<"', '">>), <<"');">>])
end.
%% F can be either a fun or a list of queries
@@ -149,758 +128,810 @@ sql_transaction(LServer, F) ->
ejabberd_odbc:sql_transaction(LServer, F).
get_last(LServer, Username) ->
- ejabberd_odbc:sql_query(
- LServer,
- ["select seconds, state from last "
- "where username='", Username, "'"]).
+ ejabberd_odbc:sql_query(LServer,
+ [<<"select seconds, state from last where "
+ "username='">>,
+ Username, <<"'">>]).
set_last_t(LServer, Username, Seconds, State) ->
- update(LServer, "last", ["username", "seconds", "state"],
+ update(LServer, <<"last">>,
+ [<<"username">>, <<"seconds">>, <<"state">>],
[Username, Seconds, State],
- ["username='", Username, "'"]).
+ [<<"username='">>, Username, <<"'">>]).
del_last(LServer, Username) ->
- ejabberd_odbc:sql_query(
- LServer,
- ["delete from last where username='", Username, "'"]).
+ ejabberd_odbc:sql_query(LServer,
+ [<<"delete from last where username='">>, Username,
+ <<"'">>]).
get_password(LServer, Username) ->
- ejabberd_odbc:sql_query(
- LServer,
- ["select password from users "
- "where username='", Username, "';"]).
+ ejabberd_odbc:sql_query(LServer,
+ [<<"select password from users where username='">>,
+ Username, <<"';">>]).
set_password_t(LServer, Username, Pass) ->
- ejabberd_odbc:sql_transaction(
- LServer,
- fun() ->
- update_t("users", ["username", "password"],
- [Username, Pass],
- ["username='", Username ,"'"])
- end).
+ ejabberd_odbc:sql_transaction(LServer,
+ fun () ->
+ update_t(<<"users">>,
+ [<<"username">>,
+ <<"password">>],
+ [Username, Pass],
+ [<<"username='">>, Username,
+ <<"'">>])
+ end).
add_user(LServer, Username, Pass) ->
- ejabberd_odbc:sql_query(
- LServer,
- ["insert into users(username, password) "
- "values ('", Username, "', '", Pass, "');"]).
+ ejabberd_odbc:sql_query(LServer,
+ [<<"insert into users(username, password) "
+ "values ('">>,
+ Username, <<"', '">>, Pass, <<"');">>]).
del_user(LServer, Username) ->
- ejabberd_odbc:sql_query(
- LServer,
- ["delete from users where username='", Username ,"';"]).
+ ejabberd_odbc:sql_query(LServer,
+ [<<"delete from users where username='">>, Username,
+ <<"';">>]).
del_user_return_password(_LServer, Username, Pass) ->
- P = ejabberd_odbc:sql_query_t(
- ["select password from users where username='",
- Username, "';"]),
- ejabberd_odbc:sql_query_t(["delete from users "
- "where username='", Username,
- "' and password='", Pass, "';"]),
+ P =
+ ejabberd_odbc:sql_query_t([<<"select password from users where username='">>,
+ Username, <<"';">>]),
+ ejabberd_odbc:sql_query_t([<<"delete from users where username='">>,
+ Username, <<"' and password='">>, Pass,
+ <<"';">>]),
P.
list_users(LServer) ->
- ejabberd_odbc:sql_query(
- LServer,
- "select username from users").
-
-list_users(LServer, [{from, Start}, {to, End}]) when is_integer(Start) and
- is_integer(End) ->
- list_users(LServer, [{limit, End-Start+1}, {offset, Start-1}]);
-list_users(LServer, [{prefix, Prefix}, {from, Start}, {to, End}]) when is_list(Prefix) and
- is_integer(Start) and
- is_integer(End) ->
- list_users(LServer, [{prefix, Prefix}, {limit, End-Start+1}, {offset, Start-1}]);
-
-list_users(LServer, [{limit, Limit}, {offset, Offset}]) when is_integer(Limit) and
- is_integer(Offset) ->
- ejabberd_odbc:sql_query(
- LServer,
- io_lib:format(
- "select username from users " ++
- "order by username " ++
- "limit ~w offset ~w", [Limit, Offset]));
-list_users(LServer, [{prefix, Prefix},
- {limit, Limit},
- {offset, Offset}]) when is_list(Prefix) and
- is_integer(Limit) and
- is_integer(Offset) ->
- ejabberd_odbc:sql_query(
- LServer,
- io_lib:format("select username from users " ++
- "where username like '~s%' " ++
- "order by username " ++
- "limit ~w offset ~w ", [Prefix, Limit, Offset])).
+ ejabberd_odbc:sql_query(LServer,
+ [<<"select username from users">>]).
+
+list_users(LServer, [{from, Start}, {to, End}])
+ when is_integer(Start) and is_integer(End) ->
+ list_users(LServer,
+ [{limit, End - Start + 1}, {offset, Start - 1}]);
+list_users(LServer,
+ [{prefix, Prefix}, {from, Start}, {to, End}])
+ when is_binary(Prefix) and is_integer(Start) and
+ is_integer(End) ->
+ list_users(LServer,
+ [{prefix, Prefix}, {limit, End - Start + 1},
+ {offset, Start - 1}]);
+list_users(LServer, [{limit, Limit}, {offset, Offset}])
+ when is_integer(Limit) and is_integer(Offset) ->
+ ejabberd_odbc:sql_query(LServer,
+ [list_to_binary(
+ io_lib:format(
+ "select username from users " ++
+ "order by username " ++
+ "limit ~w offset ~w",
+ [Limit, Offset]))]);
+list_users(LServer,
+ [{prefix, Prefix}, {limit, Limit}, {offset, Offset}])
+ when is_binary(Prefix) and is_integer(Limit) and
+ is_integer(Offset) ->
+ ejabberd_odbc:sql_query(LServer,
+ [list_to_binary(
+ io_lib:format(
+ "select username from users " ++
+ "where username like '~s%' " ++
+ "order by username " ++
+ "limit ~w offset ~w ",
+ [Prefix, Limit, Offset]))]).
users_number(LServer) ->
- case element(1, ejabberd_config:get_local_option({odbc_server, LServer})) of
- pgsql ->
- case ejabberd_config:get_local_option({pgsql_users_number_estimate, LServer}) of
- true ->
- ejabberd_odbc:sql_query(
- LServer,
- "select reltuples from pg_class where oid = 'users'::regclass::oid");
- _ ->
- ejabberd_odbc:sql_query(
- LServer,
- "select count(*) from users")
- end;
- _ ->
- ejabberd_odbc:sql_query(
- LServer,
- "select count(*) from users")
+ case element(1,
+ ejabberd_config:get_local_option(
+ {odbc_server, LServer}, fun(V) -> V end))
+ of
+ pgsql ->
+ case
+ ejabberd_config:get_local_option(
+ {pgsql_users_number_estimate, LServer},
+ fun(V) when is_boolean(V) -> V end,
+ false)
+ of
+ true ->
+ ejabberd_odbc:sql_query(LServer,
+ [<<"select reltuples from pg_class where "
+ "oid = 'users'::regclass::oid">>]);
+ _ ->
+ ejabberd_odbc:sql_query(LServer,
+ [<<"select count(*) from users">>])
+ end;
+ _ ->
+ ejabberd_odbc:sql_query(LServer,
+ [<<"select count(*) from users">>])
end.
-users_number(LServer, [{prefix, Prefix}]) when is_list(Prefix) ->
- ejabberd_odbc:sql_query(
- LServer,
- io_lib:fwrite("select count(*) from users " ++
- %% Warning: Escape prefix at higher level to prevent SQL
- %% injection.
- "where username like '~s%'", [Prefix]));
+users_number(LServer, [{prefix, Prefix}])
+ when is_binary(Prefix) ->
+ ejabberd_odbc:sql_query(LServer,
+ [list_to_binary(
+ io_lib:fwrite(
+ "select count(*) from users " ++
+ %% Warning: Escape prefix at higher level to prevent SQL
+ %% injection.
+ "where username like '~s%'",
+ [Prefix]))]);
users_number(LServer, []) ->
users_number(LServer).
add_spool_sql(Username, XML) ->
- ["insert into spool(username, xml) "
- "values ('", Username, "', '",
- XML,
- "');"].
+ [<<"insert into spool(username, xml) values ('">>,
+ Username, <<"', '">>, XML, <<"');">>].
add_spool(LServer, Queries) ->
- ejabberd_odbc:sql_transaction(
- LServer, Queries).
+ ejabberd_odbc:sql_transaction(LServer, Queries).
get_and_del_spool_msg_t(LServer, Username) ->
- F = fun() ->
- Result = ejabberd_odbc:sql_query_t(
- ["select username, xml from spool where username='", Username, "'"
- " order by seq;"]),
- ejabberd_odbc:sql_query_t(
- ["delete from spool where username='", Username, "';"]),
+ F = fun () ->
+ Result =
+ ejabberd_odbc:sql_query_t([<<"select username, xml from spool where "
+ "username='">>,
+ Username,
+ <<"' order by seq;">>]),
+ ejabberd_odbc:sql_query_t([<<"delete from spool where username='">>,
+ Username, <<"';">>]),
Result
end,
- ejabberd_odbc:sql_transaction(LServer,F).
+ ejabberd_odbc:sql_transaction(LServer, F).
del_spool_msg(LServer, Username) ->
- ejabberd_odbc:sql_query(
- LServer,
- ["delete from spool where username='", Username, "';"]).
+ ejabberd_odbc:sql_query(LServer,
+ [<<"delete from spool where username='">>, Username,
+ <<"';">>]).
get_roster(LServer, Username) ->
- ejabberd_odbc:sql_query(
- LServer,
- ["select username, jid, nick, subscription, ask, "
- "askmessage, server, subscribe, type from rosterusers "
- "where username='", Username, "'"]).
+ ejabberd_odbc:sql_query(LServer,
+ [<<"select username, jid, nick, subscription, "
+ "ask, askmessage, server, subscribe, "
+ "type from rosterusers where username='">>,
+ Username, <<"'">>]).
get_roster_jid_groups(LServer, Username) ->
- ejabberd_odbc:sql_query(
- LServer,
- ["select jid, grp from rostergroups "
- "where username='", Username, "'"]).
+ ejabberd_odbc:sql_query(LServer,
+ [<<"select jid, grp from rostergroups where "
+ "username='">>,
+ Username, <<"'">>]).
get_roster_groups(_LServer, Username, SJID) ->
- ejabberd_odbc:sql_query_t(
- ["select grp from rostergroups "
- "where username='", Username, "' "
- "and jid='", SJID, "';"]).
+ ejabberd_odbc:sql_query_t([<<"select grp from rostergroups where username='">>,
+ Username, <<"' and jid='">>, SJID, <<"';">>]).
del_user_roster_t(LServer, Username) ->
- ejabberd_odbc:sql_transaction(
- LServer,
- fun() ->
- ejabberd_odbc:sql_query_t(
- ["delete from rosterusers "
- " where username='", Username, "';"]),
- ejabberd_odbc:sql_query_t(
- ["delete from rostergroups "
- " where username='", Username, "';"])
- end).
+ ejabberd_odbc:sql_transaction(LServer,
+ fun () ->
+ ejabberd_odbc:sql_query_t([<<"delete from rosterusers where "
+ "username='">>,
+ Username,
+ <<"';">>]),
+ ejabberd_odbc:sql_query_t([<<"delete from rostergroups where "
+ "username='">>,
+ Username,
+ <<"';">>])
+ end).
get_roster_by_jid(_LServer, Username, SJID) ->
- ejabberd_odbc:sql_query_t(
- ["select username, jid, nick, subscription, "
- "ask, askmessage, server, subscribe, type from rosterusers "
- "where username='", Username, "' "
- "and jid='", SJID, "';"]).
+ ejabberd_odbc:sql_query_t([<<"select username, jid, nick, subscription, "
+ "ask, askmessage, server, subscribe, "
+ "type from rosterusers where username='">>,
+ Username, <<"' and jid='">>, SJID, <<"';">>]).
get_rostergroup_by_jid(LServer, Username, SJID) ->
- ejabberd_odbc:sql_query(
- LServer,
- ["select grp from rostergroups "
- "where username='", Username, "' "
- "and jid='", SJID, "'"]).
+ ejabberd_odbc:sql_query(LServer,
+ [<<"select grp from rostergroups where username='">>,
+ Username, <<"' and jid='">>, SJID, <<"'">>]).
del_roster(_LServer, Username, SJID) ->
- ejabberd_odbc:sql_query_t(
- ["delete from rosterusers "
- " where username='", Username, "' "
- " and jid='", SJID, "';"]),
- ejabberd_odbc:sql_query_t(
- ["delete from rostergroups "
- " where username='", Username, "' "
- " and jid='", SJID, "';"]).
+ ejabberd_odbc:sql_query_t([<<"delete from rosterusers where "
+ "username='">>,
+ Username, <<"' and jid='">>, SJID,
+ <<"';">>]),
+ ejabberd_odbc:sql_query_t([<<"delete from rostergroups where "
+ "username='">>,
+ Username, <<"' and jid='">>, SJID,
+ <<"';">>]).
del_roster_sql(Username, SJID) ->
- [["delete from rosterusers "
- " where username='", Username, "' "
- " and jid='", SJID, "';"],
- ["delete from rostergroups "
- " where username='", Username, "' "
- " and jid='", SJID, "';"]].
-
-update_roster(_LServer, Username, SJID, ItemVals, ItemGroups) ->
- update_t("rosterusers",
- ["username", "jid", "nick", "subscription", "ask",
- "askmessage", "server", "subscribe", "type"],
+ [[<<"delete from rosterusers where "
+ "username='">>,
+ Username, <<"' and jid='">>, SJID, <<"';">>],
+ [<<"delete from rostergroups where "
+ "username='">>,
+ Username, <<"' and jid='">>, SJID, <<"';">>]].
+
+update_roster(_LServer, Username, SJID, ItemVals,
+ ItemGroups) ->
+ update_t(<<"rosterusers">>,
+ [<<"username">>, <<"jid">>, <<"nick">>,
+ <<"subscription">>, <<"ask">>, <<"askmessage">>,
+ <<"server">>, <<"subscribe">>, <<"type">>],
ItemVals,
- ["username='", Username, "' and jid='", SJID, "'"]),
- ejabberd_odbc:sql_query_t(
- ["delete from rostergroups "
- " where username='", Username, "' "
- " and jid='", SJID, "';"]),
- lists:foreach(fun(ItemGroup) ->
- ejabberd_odbc:sql_query_t(
- ["insert into rostergroups("
- " username, jid, grp) "
- " values ('", join(ItemGroup, "', '"), "');"])
+ [<<"username='">>, Username, <<"' and jid='">>, SJID,
+ <<"'">>]),
+ ejabberd_odbc:sql_query_t([<<"delete from rostergroups where "
+ "username='">>,
+ Username, <<"' and jid='">>, SJID,
+ <<"';">>]),
+ lists:foreach(fun (ItemGroup) ->
+ ejabberd_odbc:sql_query_t([<<"insert into rostergroups( "
+ " username, jid, grp) values ('">>,
+ join(ItemGroup,
+ <<"', '">>),
+ <<"');">>])
end,
ItemGroups).
-update_roster_sql(Username, SJID, ItemVals, ItemGroups) ->
- [["delete from rosterusers "
- " where username='", Username, "' "
- " and jid='", SJID, "';"],
- ["insert into rosterusers("
- " username, jid, nick, "
- " subscription, ask, askmessage, "
- " server, subscribe, type) "
- " values ('", join(ItemVals, "', '"), "');"],
- ["delete from rostergroups "
- " where username='", Username, "' "
- " and jid='", SJID, "';"]] ++
- [["insert into rostergroups("
- " username, jid, grp) "
- " values ('", join(ItemGroup, "', '"), "');"] ||
- ItemGroup <- ItemGroups].
+update_roster_sql(Username, SJID, ItemVals,
+ ItemGroups) ->
+ [[<<"delete from rosterusers where "
+ "username='">>,
+ Username, <<"' and jid='">>, SJID, <<"';">>],
+ [<<"insert into rosterusers( "
+ " username, jid, nick, "
+ " subscription, ask, askmessage, "
+ " server, subscribe, type) "
+ "values ('">>,
+ join(ItemVals, <<"', '">>), <<"');">>],
+ [<<"delete from rostergroups where "
+ "username='">>,
+ Username, <<"' and jid='">>, SJID, <<"';">>]]
+ ++
+ [[<<"insert into rostergroups( "
+ " username, jid, grp) values ('">>,
+ join(ItemGroup, <<"', '">>), <<"');">>]
+ || ItemGroup <- ItemGroups].
roster_subscribe(_LServer, Username, SJID, ItemVals) ->
- update_t("rosterusers",
- ["username", "jid", "nick", "subscription", "ask",
- "askmessage", "server", "subscribe", "type"],
+ update_t(<<"rosterusers">>,
+ [<<"username">>, <<"jid">>, <<"nick">>,
+ <<"subscription">>, <<"ask">>, <<"askmessage">>,
+ <<"server">>, <<"subscribe">>, <<"type">>],
ItemVals,
- ["username='", Username, "' and jid='", SJID, "'"]).
+ [<<"username='">>, Username, <<"' and jid='">>, SJID,
+ <<"'">>]).
get_subscription(LServer, Username, SJID) ->
- ejabberd_odbc:sql_query(
- LServer,
- ["select subscription from rosterusers "
- "where username='", Username, "' "
- "and jid='", SJID, "'"]).
+ ejabberd_odbc:sql_query(LServer,
+ [<<"select subscription from rosterusers "
+ "where username='">>,
+ Username, <<"' and jid='">>, SJID, <<"'">>]).
set_private_data(_LServer, Username, LXMLNS, SData) ->
- update_t("private_storage",
- ["username", "namespace", "data"],
- [Username, LXMLNS, SData],
- ["username='", Username, "' and namespace='", LXMLNS, "'"]).
+ update_t(<<"private_storage">>,
+ [<<"username">>, <<"namespace">>, <<"data">>],
+ [Username, LXMLNS, SData],
+ [<<"username='">>, Username, <<"' and namespace='">>,
+ LXMLNS, <<"'">>]).
set_private_data_sql(Username, LXMLNS, SData) ->
- [["delete from private_storage "
- "where username='", Username, "' and "
- "namespace='", LXMLNS, "';"],
- ["insert into private_storage(username, namespace, data) "
- "values ('", Username, "', '", LXMLNS, "', "
- "'", SData, "');"]].
+ [[<<"delete from private_storage where username='">>,
+ Username, <<"' and namespace='">>, LXMLNS, <<"';">>],
+ [<<"insert into private_storage(username, "
+ "namespace, data) values ('">>,
+ Username, <<"', '">>, LXMLNS, <<"', '">>, SData,
+ <<"');">>]].
get_private_data(LServer, Username, LXMLNS) ->
- ejabberd_odbc:sql_query(
- LServer,
- ["select data from private_storage "
- "where username='", Username, "' and "
- "namespace='", LXMLNS, "';"]).
+ ejabberd_odbc:sql_query(LServer,
+ [<<"select data from private_storage where "
+ "username='">>,
+ Username, <<"' and namespace='">>, LXMLNS,
+ <<"';">>]).
+
+get_private_data(LServer, Username) ->
+ ejabberd_odbc:sql_query(LServer,
+ [<<"select namespace, data from private_storage "
+ "where username='">>, Username, <<"';">>]).
del_user_private_storage(LServer, Username) ->
- ejabberd_odbc:sql_query(
- LServer,
- ["delete from private_storage where username='", Username, "';"]).
-
-set_vcard(LServer, LUsername, SBDay, SCTRY, SEMail, SFN, SFamily, SGiven,
- SLBDay, SLCTRY, SLEMail, SLFN, SLFamily, SLGiven, SLLocality,
- SLMiddle, SLNickname, SLOrgName, SLOrgUnit, SLocality, SMiddle,
- SNickname, SOrgName, SOrgUnit, SVCARD, Username) ->
- ejabberd_odbc:sql_transaction(
- LServer,
- fun() ->
- update_t("vcard", ["username", "vcard"],
- [LUsername, SVCARD],
- ["username='", LUsername, "'"]),
- update_t("vcard_search",
- ["username", "lusername", "fn", "lfn", "family",
- "lfamily", "given", "lgiven", "middle", "lmiddle",
- "nickname", "lnickname", "bday", "lbday", "ctry",
- "lctry", "locality", "llocality", "email", "lemail",
- "orgname", "lorgname", "orgunit", "lorgunit"],
- [Username, LUsername, SFN, SLFN, SFamily, SLFamily,
- SGiven, SLGiven, SMiddle, SLMiddle, SNickname,
- SLNickname, SBDay, SLBDay, SCTRY, SLCTRY,
- SLocality, SLLocality, SEMail, SLEMail, SOrgName,
- SLOrgName, SOrgUnit, SLOrgUnit],
- ["lusername='", LUsername, "'"])
- end).
+ ejabberd_odbc:sql_query(LServer,
+ [<<"delete from private_storage where username='">>,
+ Username, <<"';">>]).
+
+set_vcard(LServer, LUsername, SBDay, SCTRY, SEMail, SFN,
+ SFamily, SGiven, SLBDay, SLCTRY, SLEMail, SLFN,
+ SLFamily, SLGiven, SLLocality, SLMiddle, SLNickname,
+ SLOrgName, SLOrgUnit, SLocality, SMiddle, SNickname,
+ SOrgName, SOrgUnit, SVCARD, Username) ->
+ ejabberd_odbc:sql_transaction(LServer,
+ fun () ->
+ update_t(<<"vcard">>,
+ [<<"username">>,
+ <<"vcard">>],
+ [LUsername, SVCARD],
+ [<<"username='">>, LUsername,
+ <<"'">>]),
+ update_t(<<"vcard_search">>,
+ [<<"username">>,
+ <<"lusername">>, <<"fn">>,
+ <<"lfn">>, <<"family">>,
+ <<"lfamily">>, <<"given">>,
+ <<"lgiven">>, <<"middle">>,
+ <<"lmiddle">>,
+ <<"nickname">>,
+ <<"lnickname">>, <<"bday">>,
+ <<"lbday">>, <<"ctry">>,
+ <<"lctry">>, <<"locality">>,
+ <<"llocality">>,
+ <<"email">>, <<"lemail">>,
+ <<"orgname">>,
+ <<"lorgname">>,
+ <<"orgunit">>,
+ <<"lorgunit">>],
+ [Username, LUsername, SFN,
+ SLFN, SFamily, SLFamily,
+ SGiven, SLGiven, SMiddle,
+ SLMiddle, SNickname,
+ SLNickname, SBDay, SLBDay,
+ SCTRY, SLCTRY, SLocality,
+ SLLocality, SEMail, SLEMail,
+ SOrgName, SLOrgName,
+ SOrgUnit, SLOrgUnit],
+ [<<"lusername='">>,
+ LUsername, <<"'">>])
+ end).
get_vcard(LServer, Username) ->
- ejabberd_odbc:sql_query(
- LServer,
- ["select vcard from vcard "
- "where username='", Username, "';"]).
+ ejabberd_odbc:sql_query(LServer,
+ [<<"select vcard from vcard where username='">>,
+ Username, <<"';">>]).
get_default_privacy_list(LServer, Username) ->
- ejabberd_odbc:sql_query(
- LServer,
- ["select name from privacy_default_list "
- "where username='", Username, "';"]).
+ ejabberd_odbc:sql_query(LServer,
+ [<<"select name from privacy_default_list "
+ "where username='">>,
+ Username, <<"';">>]).
get_default_privacy_list_t(Username) ->
- ejabberd_odbc:sql_query_t(
- ["select name from privacy_default_list "
- "where username='", Username, "';"]).
+ ejabberd_odbc:sql_query_t([<<"select name from privacy_default_list "
+ "where username='">>,
+ Username, <<"';">>]).
get_privacy_list_names(LServer, Username) ->
- ejabberd_odbc:sql_query(
- LServer,
- ["select name from privacy_list "
- "where username='", Username, "';"]).
+ ejabberd_odbc:sql_query(LServer,
+ [<<"select name from privacy_list where "
+ "username='">>,
+ Username, <<"';">>]).
get_privacy_list_names_t(Username) ->
- ejabberd_odbc:sql_query_t(
- ["select name from privacy_list "
- "where username='", Username, "';"]).
+ ejabberd_odbc:sql_query_t([<<"select name from privacy_list where "
+ "username='">>,
+ Username, <<"';">>]).
get_privacy_list_id(LServer, Username, SName) ->
- ejabberd_odbc:sql_query(
- LServer,
- ["select id from privacy_list "
- "where username='", Username, "' and name='", SName, "';"]).
+ ejabberd_odbc:sql_query(LServer,
+ [<<"select id from privacy_list where username='">>,
+ Username, <<"' and name='">>, SName, <<"';">>]).
get_privacy_list_id_t(Username, SName) ->
- ejabberd_odbc:sql_query_t(
- ["select id from privacy_list "
- "where username='", Username, "' and name='", SName, "';"]).
+ ejabberd_odbc:sql_query_t([<<"select id from privacy_list where username='">>,
+ Username, <<"' and name='">>, SName, <<"';">>]).
get_privacy_list_data(LServer, Username, SName) ->
- ejabberd_odbc:sql_query(
- LServer,
- ["select t, value, action, ord, match_all, match_iq, "
- "match_message, match_presence_in, match_presence_out "
- "from privacy_list_data "
- "where id = (select id from privacy_list where "
- " username='", Username, "' and name='", SName, "') "
- "order by ord;"]).
+ ejabberd_odbc:sql_query(LServer,
+ [<<"select t, value, action, ord, match_all, "
+ "match_iq, match_message, match_presence_in, "
+ "match_presence_out from privacy_list_data "
+ "where id = (select id from privacy_list "
+ "where username='">>,
+ Username, <<"' and name='">>, SName,
+ <<"') order by ord;">>]).
get_privacy_list_data_by_id(LServer, ID) ->
- ejabberd_odbc:sql_query(
- LServer,
- ["select t, value, action, ord, match_all, match_iq, "
- "match_message, match_presence_in, match_presence_out "
- "from privacy_list_data "
- "where id='", ID, "' order by ord;"]).
+ ejabberd_odbc:sql_query(LServer,
+ [<<"select t, value, action, ord, match_all, "
+ "match_iq, match_message, match_presence_in, "
+ "match_presence_out from privacy_list_data "
+ "where id='">>,
+ ID, <<"' order by ord;">>]).
get_privacy_list_data_by_id_t(ID) ->
- ejabberd_odbc:sql_query_t(
- ["select t, value, action, ord, match_all, match_iq, "
- "match_message, match_presence_in, match_presence_out "
- "from privacy_list_data "
- "where id='", ID, "' order by ord;"]).
+ ejabberd_odbc:sql_query_t([<<"select t, value, action, ord, match_all, "
+ "match_iq, match_message, match_presence_in, "
+ "match_presence_out from privacy_list_data "
+ "where id='">>,
+ ID, <<"' order by ord;">>]).
set_default_privacy_list(Username, SName) ->
- update_t("privacy_default_list", ["username", "name"],
- [Username, SName], ["username='", Username, "'"]).
+ update_t(<<"privacy_default_list">>,
+ [<<"username">>, <<"name">>], [Username, SName],
+ [<<"username='">>, Username, <<"'">>]).
unset_default_privacy_list(LServer, Username) ->
- ejabberd_odbc:sql_query(
- LServer,
- ["delete from privacy_default_list "
- " where username='", Username, "';"]).
+ ejabberd_odbc:sql_query(LServer,
+ [<<"delete from privacy_default_list "
+ " where username='">>,
+ Username, <<"';">>]).
remove_privacy_list(Username, SName) ->
- ejabberd_odbc:sql_query_t(
- ["delete from privacy_list "
- "where username='", Username, "' and name='", SName, "';"]).
+ ejabberd_odbc:sql_query_t([<<"delete from privacy_list where username='">>,
+ Username, <<"' and name='">>, SName, <<"';">>]).
add_privacy_list(Username, SName) ->
- ejabberd_odbc:sql_query_t(
- ["insert into privacy_list(username, name) "
- "values ('", Username, "', '", SName, "');"]).
+ ejabberd_odbc:sql_query_t([<<"insert into privacy_list(username, name) "
+ "values ('">>,
+ Username, <<"', '">>, SName, <<"');">>]).
set_privacy_list(ID, RItems) ->
- ejabberd_odbc:sql_query_t(
- ["delete from privacy_list_data "
- "where id='", ID, "';"]),
- lists:foreach(fun(Items) ->
- ejabberd_odbc:sql_query_t(
- ["insert into privacy_list_data("
- "id, t, value, action, ord, match_all, match_iq, "
- "match_message, match_presence_in, "
- "match_presence_out "
- ") "
- "values ('", ID, "', '",
- join(Items, "', '"), "');"])
- end, RItems).
+ ejabberd_odbc:sql_query_t([<<"delete from privacy_list_data where "
+ "id='">>,
+ ID, <<"';">>]),
+ lists:foreach(fun (Items) ->
+ ejabberd_odbc:sql_query_t([<<"insert into privacy_list_data(id, t, "
+ "value, action, ord, match_all, match_iq, "
+ "match_message, match_presence_in, match_prese"
+ "nce_out ) values ('">>,
+ ID, <<"', '">>,
+ join(Items, <<"', '">>),
+ <<"');">>])
+ end,
+ RItems).
del_privacy_lists(LServer, Server, Username) ->
- ejabberd_odbc:sql_query(
- LServer,
- ["delete from privacy_list where username='", Username, "';"]),
- ejabberd_odbc:sql_query(
- LServer,
- ["delete from privacy_list_data where value='", Username++"@"++Server, "';"]),
- ejabberd_odbc:sql_query(
- LServer,
- ["delete from privacy_default_list where username='", Username, "';"]).
-
%% Characters to escape
-escape($\0) -> "\\0";
-escape($\n) -> "\\n";
-escape($\t) -> "\\t";
-escape($\b) -> "\\b";
-escape($\r) -> "\\r";
-escape($') -> "''";
-escape($") -> "\\\"";
-escape($\\) -> "\\\\";
-escape(C) -> C.
-
%% Count number of records in a table given a where clause
-count_records_where(LServer, Table, WhereClause) ->
- ejabberd_odbc:sql_query(
- LServer,
- ["select count(*) from ", Table, " ", WhereClause, ";"]).
+ ejabberd_odbc:sql_query(LServer,
+ [<<"delete from privacy_list where username='">>,
+ Username, <<"';">>]),
+ ejabberd_odbc:sql_query(LServer,
+ [<<"delete from privacy_list_data where "
+ "value='">>,
+ <<Username/binary, "@", Server/binary>>,
+ <<"';">>]),
+ ejabberd_odbc:sql_query(LServer,
+ [<<"delete from privacy_default_list where "
+ "username='">>,
+ Username, <<"';">>]).
+
+escape($\000) -> <<"\\0">>;
+escape($\n) -> <<"\\n">>;
+escape($\t) -> <<"\\t">>;
+escape($\b) -> <<"\\b">>;
+escape($\r) -> <<"\\r">>;
+escape($') -> <<"''">>;
+escape($") -> <<"\\\"">>;
+escape($\\) -> <<"\\\\">>;
+escape(C) -> <<C>>.
+count_records_where(LServer, Table, WhereClause) ->
+ ejabberd_odbc:sql_query(LServer,
+ [<<"select count(*) from ">>, Table, <<" ">>,
+ WhereClause, <<";">>]).
get_roster_version(LServer, LUser) ->
- ejabberd_odbc:sql_query(LServer,
- ["select version from roster_version where username = '", LUser, "'"]).
+ ejabberd_odbc:sql_query(LServer,
+ [<<"select version from roster_version where "
+ "username = '">>,
+ LUser, <<"'">>]).
+
set_roster_version(LUser, Version) ->
- update_t("roster_version", ["username", "version"], [LUser, Version], ["username = '", LUser, "'"]).
+ update_t(<<"roster_version">>,
+ [<<"username">>, <<"version">>], [LUser, Version],
+ [<<"username = '">>, LUser, <<"'">>]).
+
-endif.
%% -----------------
%% MSSQL queries
-ifdef(mssql).
-get_db_type() ->
- mssql.
-
%% Queries can be either a fun or a list of queries
-sql_transaction(LServer, Queries) when is_list(Queries) ->
- %% SQL transaction based on a list of queries
- %% This function automatically
- F = fun() ->
- lists:foreach(fun(Query) ->
- ejabberd_odbc:sql_query(LServer, Query)
- end, Queries)
- end,
+get_db_type() -> mssql.
+
+sql_transaction(LServer, Queries)
+ when is_list(Queries) ->
+ F = fun () ->
+ lists:foreach(fun (Query) ->
+ ejabberd_odbc:sql_query(LServer, Query)
+ end,
+ Queries)
+ end,
{atomic, catch F()};
sql_transaction(_LServer, FQueries) ->
{atomic, catch FQueries()}.
get_last(LServer, Username) ->
- ejabberd_odbc:sql_query(
- LServer,
- ["EXECUTE dbo.get_last '", Username, "'"]).
+ ejabberd_odbc:sql_query(LServer,
+ [<<"EXECUTE dbo.get_last '">>, Username, <<"'">>]).
set_last_t(LServer, Username, Seconds, State) ->
- Result = ejabberd_odbc:sql_query(
- LServer,
- ["EXECUTE dbo.set_last '", Username, "', '", Seconds,
- "', '", State, "'"]),
+ Result = ejabberd_odbc:sql_query(LServer,
+ [<<"EXECUTE dbo.set_last '">>, Username,
+ <<"', '">>, Seconds, <<"', '">>, State,
+ <<"'">>]),
{atomic, Result}.
del_last(LServer, Username) ->
- ejabberd_odbc:sql_query(
- LServer,
- ["EXECUTE dbo.del_last '", Username, "'"]).
+ ejabberd_odbc:sql_query(LServer,
+ [<<"EXECUTE dbo.del_last '">>, Username, <<"'">>]).
get_password(LServer, Username) ->
- ejabberd_odbc:sql_query(
- LServer,
- ["EXECUTE dbo.get_password '", Username, "'"]).
+ ejabberd_odbc:sql_query(LServer,
+ [<<"EXECUTE dbo.get_password '">>, Username,
+ <<"'">>]).
set_password_t(LServer, Username, Pass) ->
- Result = ejabberd_odbc:sql_query(
- LServer,
- ["EXECUTE dbo.set_password '", Username, "', '", Pass, "'"]),
+ Result = ejabberd_odbc:sql_query(LServer,
+ [<<"EXECUTE dbo.set_password '">>,
+ Username, <<"', '">>, Pass, <<"'">>]),
{atomic, Result}.
add_user(LServer, Username, Pass) ->
- ejabberd_odbc:sql_query(
- LServer,
- ["EXECUTE dbo.add_user '", Username, "', '", Pass, "'"]).
+ ejabberd_odbc:sql_query(LServer,
+ [<<"EXECUTE dbo.add_user '">>, Username, <<"', '">>,
+ Pass, <<"'">>]).
del_user(LServer, Username) ->
- ejabberd_odbc:sql_query(
- LServer,
- ["EXECUTE dbo.del_user '", Username ,"'"]).
+ ejabberd_odbc:sql_query(LServer,
+ [<<"EXECUTE dbo.del_user '">>, Username, <<"'">>]).
del_user_return_password(LServer, Username, Pass) ->
- ejabberd_odbc:sql_query(
- LServer,
- ["EXECUTE dbo.del_user_return_password '", Username, "'"]),
+ ejabberd_odbc:sql_query(LServer,
+ [<<"EXECUTE dbo.del_user_return_password '">>,
+ Username, <<"'">>]),
Pass.
list_users(LServer) ->
- ejabberd_odbc:sql_query(
- LServer,
- "EXECUTE dbo.list_users").
+ ejabberd_odbc:sql_query(LServer,
+ <<"EXECUTE dbo.list_users">>).
-list_users(LServer, _) ->
- % scope listing not supported
- list_users(LServer).
+list_users(LServer, _) -> list_users(LServer).
users_number(LServer) ->
- ejabberd_odbc:sql_query(
- LServer,
- "select count(*) from users with (nolock)").
+ ejabberd_odbc:sql_query(LServer,
+ <<"select count(*) from users with (nolock)">>).
-users_number(LServer, _) ->
- % scope listing not supported
- users_number(LServer).
+users_number(LServer, _) -> users_number(LServer).
add_spool_sql(Username, XML) ->
- ["EXECUTE dbo.add_spool '", Username, "' , '",XML,"'"].
+ [<<"EXECUTE dbo.add_spool '">>, Username, <<"' , '">>,
+ XML, <<"'">>].
add_spool(LServer, Queries) ->
- lists:foreach(fun(Query) ->
+ lists:foreach(fun (Query) ->
ejabberd_odbc:sql_query(LServer, Query)
end,
Queries).
get_and_del_spool_msg_t(LServer, Username) ->
- [Result] = case ejabberd_odbc:sql_query(
- LServer,
- ["EXECUTE dbo.get_and_del_spool_msg '", Username, "'"]) of
- Rs when is_list(Rs) ->
- lists:filter(fun({selected, _Header, _Row}) ->
- true;
- ({updated, _N}) ->
- false
+ [Result] = case ejabberd_odbc:sql_query(LServer,
+ [<<"EXECUTE dbo.get_and_del_spool_msg '">>,
+ Username, <<"'">>])
+ of
+ Rs when is_list(Rs) ->
+ lists:filter(fun ({selected, _Header, _Row}) -> true;
+ ({updated, _N}) -> false
end,
Rs);
- Rs -> [Rs]
+ Rs -> [Rs]
end,
{atomic, Result}.
del_spool_msg(LServer, Username) ->
- ejabberd_odbc:sql_query(
- LServer,
- ["EXECUTE dbo.del_spool_msg '", Username, "'"]).
+ ejabberd_odbc:sql_query(LServer,
+ [<<"EXECUTE dbo.del_spool_msg '">>, Username,
+ <<"'">>]).
get_roster(LServer, Username) ->
- ejabberd_odbc:sql_query(
- LServer,
- ["EXECUTE dbo.get_roster '", Username, "'"]).
+ ejabberd_odbc:sql_query(LServer,
+ [<<"EXECUTE dbo.get_roster '">>, Username,
+ <<"'">>]).
get_roster_jid_groups(LServer, Username) ->
- ejabberd_odbc:sql_query(
- LServer,
- ["EXECUTE dbo.get_roster_jid_groups '", Username, "'"]).
+ ejabberd_odbc:sql_query(LServer,
+ [<<"EXECUTE dbo.get_roster_jid_groups '">>,
+ Username, <<"'">>]).
get_roster_groups(LServer, Username, SJID) ->
- ejabberd_odbc:sql_query(
- LServer,
- ["EXECUTE dbo.get_roster_groups '", Username, "' , '", SJID, "'"]).
+ ejabberd_odbc:sql_query(LServer,
+ [<<"EXECUTE dbo.get_roster_groups '">>, Username,
+ <<"' , '">>, SJID, <<"'">>]).
del_user_roster_t(LServer, Username) ->
- Result = ejabberd_odbc:sql_query(
- LServer,
- ["EXECUTE dbo.del_user_roster '", Username, "'"]),
+ Result = ejabberd_odbc:sql_query(LServer,
+ [<<"EXECUTE dbo.del_user_roster '">>,
+ Username, <<"'">>]),
{atomic, Result}.
get_roster_by_jid(LServer, Username, SJID) ->
- ejabberd_odbc:sql_query(
- LServer,
- ["EXECUTE dbo.get_roster_by_jid '", Username, "' , '", SJID, "'"]).
+ ejabberd_odbc:sql_query(LServer,
+ [<<"EXECUTE dbo.get_roster_by_jid '">>, Username,
+ <<"' , '">>, SJID, <<"'">>]).
get_rostergroup_by_jid(LServer, Username, SJID) ->
- ejabberd_odbc:sql_query(
- LServer,
- ["EXECUTE dbo.get_rostergroup_by_jid '", Username, "' , '", SJID, "'"]).
+ ejabberd_odbc:sql_query(LServer,
+ [<<"EXECUTE dbo.get_rostergroup_by_jid '">>,
+ Username, <<"' , '">>, SJID, <<"'">>]).
del_roster(LServer, Username, SJID) ->
- ejabberd_odbc:sql_query(
- LServer,
- ["EXECUTE dbo.del_roster '", Username, "', '", SJID, "'"]).
+ ejabberd_odbc:sql_query(LServer,
+ [<<"EXECUTE dbo.del_roster '">>, Username,
+ <<"', '">>, SJID, <<"'">>]).
del_roster_sql(Username, SJID) ->
- ["EXECUTE dbo.del_roster '", Username, "', '", SJID, "'"].
+ [<<"EXECUTE dbo.del_roster '">>, Username, <<"', '">>,
+ SJID, <<"'">>].
-update_roster(LServer, Username, SJID, ItemVals, ItemGroups) ->
- Query1 = ["EXECUTE dbo.del_roster '", Username, "', '", SJID, "' "],
+update_roster(LServer, Username, SJID, ItemVals,
+ ItemGroups) ->
+ Query1 = [<<"EXECUTE dbo.del_roster '">>, Username,
+ <<"', '">>, SJID, <<"' ">>],
ejabberd_odbc:sql_query(LServer, lists:flatten(Query1)),
- Query2 = ["EXECUTE dbo.add_roster_user ", ItemVals],
+ Query2 = [<<"EXECUTE dbo.add_roster_user ">>, ItemVals],
ejabberd_odbc:sql_query(LServer, lists:flatten(Query2)),
- Query3 = ["EXECUTE dbo.del_roster_groups '", Username, "', '", SJID, "' "],
+ Query3 = [<<"EXECUTE dbo.del_roster_groups '">>,
+ Username, <<"', '">>, SJID, <<"' ">>],
ejabberd_odbc:sql_query(LServer, lists:flatten(Query3)),
- lists:foreach(fun(ItemGroup) ->
- Query = ["EXECUTE dbo.add_roster_group ",
+ lists:foreach(fun (ItemGroup) ->
+ Query = [<<"EXECUTE dbo.add_roster_group ">>,
ItemGroup],
- ejabberd_odbc:sql_query(LServer,
- lists:flatten(Query))
+ ejabberd_odbc:sql_query(LServer, lists:flatten(Query))
end,
ItemGroups).
-update_roster_sql(Username, SJID, ItemVals, ItemGroups) ->
- ["BEGIN TRANSACTION ",
- "EXECUTE dbo.del_roster_groups '", Username, "','", SJID, "' ",
- "EXECUTE dbo.add_roster_user ", ItemVals, " "] ++
- [lists:flatten("EXECUTE dbo.add_roster_group ", ItemGroup, " ")
- || ItemGroup <- ItemGroups] ++
- ["COMMIT"].
+update_roster_sql(Username, SJID, ItemVals,
+ ItemGroups) ->
+ [<<"BEGIN TRANSACTION ">>,
+ <<"EXECUTE dbo.del_roster_groups '">>, Username,
+ <<"','">>, SJID, <<"' ">>,
+ <<"EXECUTE dbo.add_roster_user ">>, ItemVals, <<" ">>]
+ ++
+ [lists:flatten(<<"EXECUTE dbo.add_roster_group ">>,
+ ItemGroup, <<" ">>)
+ || ItemGroup <- ItemGroups]
+ ++ [<<"COMMIT">>].
roster_subscribe(LServer, _Username, _SJID, ItemVals) ->
- catch ejabberd_odbc:sql_query(
- LServer,
- ["EXECUTE dbo.add_roster_user ", ItemVals]).
+ catch ejabberd_odbc:sql_query(LServer,
+ [<<"EXECUTE dbo.add_roster_user ">>,
+ ItemVals]).
get_subscription(LServer, Username, SJID) ->
- ejabberd_odbc:sql_query(
- LServer,
- ["EXECUTE dbo.get_subscription '", Username, "' , '", SJID, "'"]).
+ ejabberd_odbc:sql_query(LServer,
+ [<<"EXECUTE dbo.get_subscription '">>, Username,
+ <<"' , '">>, SJID, <<"'">>]).
set_private_data(LServer, Username, LXMLNS, SData) ->
- ejabberd_odbc:sql_query(
- LServer,
- set_private_data_sql(Username, LXMLNS, SData)).
+ ejabberd_odbc:sql_query(LServer,
+ set_private_data_sql(Username, LXMLNS, SData)).
set_private_data_sql(Username, LXMLNS, SData) ->
- ["EXECUTE dbo.set_private_data '", Username, "' , '", LXMLNS, "' , '", SData, "'"].
+ [<<"EXECUTE dbo.set_private_data '">>, Username,
+ <<"' , '">>, LXMLNS, <<"' , '">>, SData, <<"'">>].
get_private_data(LServer, Username, LXMLNS) ->
- ejabberd_odbc:sql_query(
- LServer,
- ["EXECUTE dbo.get_private_data '", Username, "' , '", LXMLNS, "'"]).
+ ejabberd_odbc:sql_query(LServer,
+ [<<"EXECUTE dbo.get_private_data '">>, Username,
+ <<"' , '">>, LXMLNS, <<"'">>]).
del_user_private_storage(LServer, Username) ->
- ejabberd_odbc:sql_query(
- LServer,
- ["EXECUTE dbo.del_user_storage '", Username, "'"]).
-
-set_vcard(LServer, LUsername, SBDay, SCTRY, SEMail, SFN, SFamily, SGiven,
- SLBDay, SLCTRY, SLEMail, SLFN, SLFamily, SLGiven, SLLocality,
- SLMiddle, SLNickname, SLOrgName, SLOrgUnit, SLocality, SMiddle,
- SNickname, SOrgName, SOrgUnit, SVCARD, Username) ->
- ejabberd_odbc:sql_query(
- LServer,
- ["EXECUTE dbo.set_vcard '", SVCARD, "' , '", Username, "' , '", LUsername, "' , '",
- SFN, "' , '", SLFN, "' , '", SFamily, "' , '", SLFamily, "' , '",
- SGiven, "' , '", SLGiven, "' , '", SMiddle, "' , '", SLMiddle, "' , '",
- SNickname, "' , '", SLNickname, "' , '", SBDay, "' , '", SLBDay, "' , '",
- SCTRY, "' , '", SLCTRY, "' , '", SLocality, "' , '", SLLocality, "' , '",
- SEMail, "' , '", SLEMail, "' , '", SOrgName, "' , '", SLOrgName, "' , '",
- SOrgUnit, "' , '", SLOrgUnit, "'"]).
+ ejabberd_odbc:sql_query(LServer,
+ [<<"EXECUTE dbo.del_user_storage '">>, Username,
+ <<"'">>]).
+
+set_vcard(LServer, LUsername, SBDay, SCTRY, SEMail, SFN,
+ SFamily, SGiven, SLBDay, SLCTRY, SLEMail, SLFN,
+ SLFamily, SLGiven, SLLocality, SLMiddle, SLNickname,
+ SLOrgName, SLOrgUnit, SLocality, SMiddle, SNickname,
+ SOrgName, SOrgUnit, SVCARD, Username) ->
+ ejabberd_odbc:sql_query(LServer,
+ [<<"EXECUTE dbo.set_vcard '">>, SVCARD, <<"' , '">>,
+ Username, <<"' , '">>, LUsername, <<"' , '">>, SFN,
+ <<"' , '">>, SLFN, <<"' , '">>, SFamily,
+ <<"' , '">>, SLFamily, <<"' , '">>, SGiven,
+ <<"' , '">>, SLGiven, <<"' , '">>, SMiddle,
+ <<"' , '">>, SLMiddle, <<"' , '">>, SNickname,
+ <<"' , '">>, SLNickname, <<"' , '">>, SBDay,
+ <<"' , '">>, SLBDay, <<"' , '">>, SCTRY,
+ <<"' , '">>, SLCTRY, <<"' , '">>, SLocality,
+ <<"' , '">>, SLLocality, <<"' , '">>, SEMail,
+ <<"' , '">>, SLEMail, <<"' , '">>, SOrgName,
+ <<"' , '">>, SLOrgName, <<"' , '">>, SOrgUnit,
+ <<"' , '">>, SLOrgUnit, <<"'">>]).
get_vcard(LServer, Username) ->
- ejabberd_odbc:sql_query(
- LServer,
- ["EXECUTE dbo.get_vcard '", Username, "'"]).
+ ejabberd_odbc:sql_query(LServer,
+ [<<"EXECUTE dbo.get_vcard '">>, Username, <<"'">>]).
get_default_privacy_list(LServer, Username) ->
- ejabberd_odbc:sql_query(
- LServer,
- ["EXECUTE dbo.get_default_privacy_list '", Username, "'"]).
+ ejabberd_odbc:sql_query(LServer,
+ [<<"EXECUTE dbo.get_default_privacy_list '">>,
+ Username, <<"'">>]).
get_default_privacy_list_t(Username) ->
- ejabberd_odbc:sql_query_t(
- ["EXECUTE dbo.get_default_privacy_list '", Username, "'"]).
+ ejabberd_odbc:sql_query_t([<<"EXECUTE dbo.get_default_privacy_list '">>,
+ Username, <<"'">>]).
get_privacy_list_names(LServer, Username) ->
- ejabberd_odbc:sql_query(
- LServer,
- ["EXECUTE dbo.get_privacy_list_names '", Username, "'"]).
+ ejabberd_odbc:sql_query(LServer,
+ [<<"EXECUTE dbo.get_privacy_list_names '">>,
+ Username, <<"'">>]).
get_privacy_list_names_t(Username) ->
- ejabberd_odbc:sql_query_t(
- ["EXECUTE dbo.get_privacy_list_names '", Username, "'"]).
+ ejabberd_odbc:sql_query_t([<<"EXECUTE dbo.get_privacy_list_names '">>,
+ Username, <<"'">>]).
get_privacy_list_id(LServer, Username, SName) ->
- ejabberd_odbc:sql_query(
- LServer,
- ["EXECUTE dbo.get_privacy_list_id '", Username, "' , '", SName, "'"]).
+ ejabberd_odbc:sql_query(LServer,
+ [<<"EXECUTE dbo.get_privacy_list_id '">>, Username,
+ <<"' , '">>, SName, <<"'">>]).
get_privacy_list_id_t(Username, SName) ->
- ejabberd_odbc:sql_query_t(
- ["EXECUTE dbo.get_privacy_list_id '", Username, "' , '", SName, "'"]).
+ ejabberd_odbc:sql_query_t([<<"EXECUTE dbo.get_privacy_list_id '">>,
+ Username, <<"' , '">>, SName, <<"'">>]).
get_privacy_list_data(LServer, Username, SName) ->
- ejabberd_odbc:sql_query(
- LServer,
- ["EXECUTE dbo.get_privacy_list_data '", Username, "' , '", SName, "'"]).
+ ejabberd_odbc:sql_query(LServer,
+ [<<"EXECUTE dbo.get_privacy_list_data '">>,
+ Username, <<"' , '">>, SName, <<"'">>]).
get_privacy_list_data_by_id(LServer, ID) ->
- ejabberd_odbc:sql_query(
- LServer,
- ["EXECUTE dbo.get_privacy_list_data_by_id '", ID, "'"]).
+ ejabberd_odbc:sql_query(LServer,
+ [<<"EXECUTE dbo.get_privacy_list_data_by_id '">>,
+ ID, <<"'">>]).
get_privacy_list_data_by_id_t(ID) ->
- ejabberd_odbc:sql_query_t(
- ["EXECUTE dbo.get_privacy_list_data_by_id '", ID, "'"]).
+ ejabberd_odbc:sql_query_t([<<"EXECUTE dbo.get_privacy_list_data_by_id '">>,
+ ID, <<"'">>]).
set_default_privacy_list(Username, SName) ->
- ejabberd_odbc:sql_query_t(
- ["EXECUTE dbo.set_default_privacy_list '", Username, "' , '", SName, "'"]).
+ ejabberd_odbc:sql_query_t([<<"EXECUTE dbo.set_default_privacy_list '">>,
+ Username, <<"' , '">>, SName, <<"'">>]).
unset_default_privacy_list(LServer, Username) ->
- ejabberd_odbc:sql_query(
- LServer,
- ["EXECUTE dbo.unset_default_privacy_list '", Username, "'"]).
+ ejabberd_odbc:sql_query(LServer,
+ [<<"EXECUTE dbo.unset_default_privacy_list '">>,
+ Username, <<"'">>]).
remove_privacy_list(Username, SName) ->
- ejabberd_odbc:sql_query_t(
- ["EXECUTE dbo.remove_privacy_list '", Username, "' , '", SName, "'"]).
+ ejabberd_odbc:sql_query_t([<<"EXECUTE dbo.remove_privacy_list '">>,
+ Username, <<"' , '">>, SName, <<"'">>]).
add_privacy_list(Username, SName) ->
- ejabberd_odbc:sql_query_t(
- ["EXECUTE dbo.add_privacy_list '", Username, "' , '", SName, "'"]).
+ ejabberd_odbc:sql_query_t([<<"EXECUTE dbo.add_privacy_list '">>,
+ Username, <<"' , '">>, SName, <<"'">>]).
set_privacy_list(ID, RItems) ->
- ejabberd_odbc:sql_query_t(
- ["EXECUTE dbo.del_privacy_list_by_id '", ID, "'"]),
-
- lists:foreach(fun(Items) ->
- ejabberd_odbc:sql_query_t(
- ["EXECUTE dbo.set_privacy_list '", ID, "', '", join(Items, "', '"), "'"])
- end, RItems).
+ ejabberd_odbc:sql_query_t([<<"EXECUTE dbo.del_privacy_list_by_id '">>,
+ ID, <<"'">>]),
+ lists:foreach(fun (Items) ->
+ ejabberd_odbc:sql_query_t([<<"EXECUTE dbo.set_privacy_list '">>,
+ ID, <<"', '">>,
+ join(Items, <<"', '">>),
+ <<"'">>])
+ end,
+ RItems).
del_privacy_lists(LServer, Server, Username) ->
- ejabberd_odbc:sql_query(
- LServer,
- ["EXECUTE dbo.del_privacy_lists @Server='", Server ,"' @username='", Username, "'"]).
-
%% Characters to escape
-escape($\0) -> "\\0";
-escape($\t) -> "\\t";
-escape($\b) -> "\\b";
-escape($\r) -> "\\r";
-escape($') -> "\''";
-escape($") -> "\\\"";
-escape(C) -> C.
-
%% Count number of records in a table given a where clause
+ ejabberd_odbc:sql_query(LServer,
+ [<<"EXECUTE dbo.del_privacy_lists @Server='">>,
+ Server, <<"' @username='">>, Username, <<"'">>]).
+
+escape($\000) -> <<"\\0">>;
+escape($\t) -> <<"\\t">>;
+escape($\b) -> <<"\\b">>;
+escape($\r) -> <<"\\r">>;
+escape($') -> <<"''">>;
+escape($") -> <<"\\\"">>;
+escape(C) -> C.
+
count_records_where(LServer, Table, WhereClause) ->
- ejabberd_odbc:sql_query(
- LServer,
- ["select count(*) from ", Table, " with (nolock) ", WhereClause]).
+ ejabberd_odbc:sql_query(LServer,
+ [<<"select count(*) from ">>, Table,
+ <<" with (nolock) ">>, WhereClause]).
get_roster_version(LServer, LUser) ->
- ejabberd_odbc:sql_query(
- LServer,
- ["EXECUTE dbo.get_roster_version '", LUser, "'"]).
+ ejabberd_odbc:sql_query(LServer,
+ [<<"EXECUTE dbo.get_roster_version '">>, LUser,
+ <<"'">>]).
set_roster_version(Username, Version) ->
- %% This function doesn't know the vhost, so we hope it's the first one defined:
- LServer = ?MYNAME,
- ejabberd_odbc:sql_query(
- LServer,
- ["EXECUTE dbo.set_roster_version '", Username, "', '", Version, "'"]).
+ LServer = (?MYNAME),
+ ejabberd_odbc:sql_query(LServer,
+ [<<"EXECUTE dbo.set_roster_version '">>, Username,
+ <<"', '">>, Version, <<"'">>]).
+
-endif.
diff --git a/src/odbc/pg.sql b/src/odbc/pg.sql
index 4ed897e79..0b641d575 100644
--- a/src/odbc/pg.sql
+++ b/src/odbc/pg.sql
@@ -274,3 +274,12 @@ CREATE TABLE motd (
xml text,
created_at TIMESTAMP NOT NULL DEFAULT now()
);
+
+CREATE TABLE caps_features (
+ node text NOT NULL,
+ subnode text NOT NULL,
+ feature text,
+ created_at TIMESTAMP NOT NULL DEFAULT now()
+);
+
+CREATE INDEX i_caps_features_node_subnode ON caps_features USING btree (node, subnode);