1 /* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
2 /* vim:set ts=2 sw=2 sts=2 et cindent tw=79 ft=cpp: */
4 * Copyright (C) 2007,2008 Sergey Yanovich <ynvich@gmail.com>
6 * This program is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU General Public License as
8 * published by the Free Software Foundation; either version 2 of the
9 * License, or (at your option) any later version.
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * General Public License for more details.
16 * You should have received a copy of the GNU General Public
17 * License along with this program; if not, write to the
18 * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
19 * Boston, MA 02111-1307, USA.
22 #include "xpcom-config.h"
25 #include "nsComponentManagerUtils.h"
26 #include "nsServiceManagerUtils.h"
27 #include "nsIIOService.h"
30 #include "nsStringAPI.h"
32 #include "nsDebugUtils.h"
35 #include "nsAppDirectoryServiceDefs.h"
36 #include "mozStorageCID.h"
37 #include "mozIStorageService.h"
38 #include "mozIStorageConnection.h"
40 /* Project includes */
41 #include "aaIManager.h"
42 #include "aaISqlRequest.h"
43 #include "aaISqlTransaction.h"
44 #include "aaSessionUtils.h"
45 #include "aaBaseLoaders.h"
46 #include "aaSqlChannel.h"
47 #include "aaSession.h"
49 /******** Transaction Guard ********/
53 aaGuard(mozIStorageConnection
*aConnection
)
54 :mConnection(aConnection
) {;}
55 ~aaGuard() { if (mConnection
) mConnection
->RollbackTransaction(); }
58 mozIStorageConnection
*mConnection
;
64 mozIStorageConnection
*tmp
= mConnection
;
66 return tmp
->CommitTransaction();
69 /******** aaSession ********/
70 aaSession::aaSession()
74 aaSession::~aaSession()
78 NS_IMPL_ISUPPORTS2(aaSession
,
84 aaSession::GetConnectionURI(nsIURI
* * _retval
)
86 NS_ENSURE_ARG_POINTER(_retval
);
89 nsCOMPtr
<nsIIOService
> ios
= do_GetService(NS_IOSERVICE_CONTRACTID
, &rv
);
90 NS_ENSURE_SUCCESS(rv
, rv
);
92 nsCOMPtr
<nsIFile
> file
;
93 rv
= mConnection
->GetDatabaseFile(getter_AddRefs(file
));
94 NS_ENSURE_SUCCESS(rv
, rv
);
96 rv
= ios
->NewFileURI(file
, _retval
);
97 NS_ENSURE_SUCCESS(rv
, rv
);
103 aaSession::Load(aaISqlRequest
*aRequest
, nsIArray
* * _retval
)
105 NS_ENSURE_ARG_POINTER(_retval
);
108 mLoader
= do_CreateInstance(AA_SQLSIMPLELOADER_CONTRACT
, &rv
);
109 NS_ENSURE_SUCCESS(rv
, rv
);
111 nsCOMPtr
<nsISupports
> param
= do_QueryInterface(aRequest
);
112 rv
= mLoader
->SetParam(param
);
113 NS_ENSURE_SUCCESS(rv
, rv
);
114 nsCOMPtr
<nsISupports
> retval
;
115 rv
= Execute(mLoader
, getter_AddRefs(retval
));
116 NS_ENSURE_SUCCESS(rv
, rv
);
117 return CallQueryInterface(retval
, _retval
);
121 aaSession::Execute(aaISqlTransaction
*aTransaction
, nsISupports
* * _retval
)
123 NS_ENSURE_ARG_POINTER(aTransaction
);
124 NS_ENSURE_STATE(mConnection
);
127 nsRefPtr
<aaSqlChannel
> channel
= new aaSqlChannel(this);
128 NS_ENSURE_TRUE(channel
, NS_ERROR_OUT_OF_MEMORY
);
130 rv
= mConnection
->BeginTransaction();
131 NS_ENSURE_SUCCESS(rv
, rv
);
133 aaGuard
guard(mConnection
);
134 rv
= aTransaction
->Execute(channel
, _retval
);
135 if (NS_FAILED(rv
)) return rv
;
137 return guard
.Dismiss();
142 aaSession::Save(aaIDataNode
*aNode
, aaIDataNode
*aOldNode
)
146 mDispatcher
= do_CreateInstance("@aasii.org/storage/save-dispatcher;1",
148 NS_ENSURE_SUCCESS(rv
, rv
);
150 return mDispatcher
->Save(aNode
, aOldNode
);
155 /* Private methods */
161 const struct aaTable tables
[] =
164 "id INTEGER PRIMARY KEY AUTOINCREMENT,\
165 tag CHAR (47) UNIQUE"}
168 "type INTEGER NOT NULL, id INTEGER NOT NULL,\
169 PRIMARY KEY (type, id)"}
172 "id INTEGER PRIMARY KEY AUTOINCREMENT,\
173 tag CHAR (47) UNIQUE"}
176 "id INTEGER PRIMARY KEY NOT NULL,\
177 alpha_code CHAR (3) UNIQUE NOT NULL"}
180 "id INTEGER PRIMARY KEY AUTOINCREMENT,\
181 tag CHAR (47) UNIQUE, entity_id INTEGER NOT NULL, rate DOUBLE NOT NULL,\
182 off_balance BOOLEAN NOT NULL"}
185 "flow_id INTEGER NOT NULL, side BOOLEAN NOT\
186 NULL, resource_type INTEGER NOT NULL, resource_id INTEGER NOT NULL,\
187 PRIMARY KEY (flow_id, side)"}
190 "day INTEGER NOT NULL, event_id INTEGER NOT NULL,\
191 transfer_id INTEGER NOT NULL, side BOOLEAN NOT NULL, flow_id INTEGER\
192 NOT NULL, PRIMARY KEY (day, event_id, transfer_id, side)"}
195 "flow_id INTEGER NOT NULL,\
196 id INTEGER NOT NULL,\
197 fact_side BOOLEAN NOT NULL,\
198 change_side BOOLEAN NOT NULL,\
199 rate DOUBLE NOT NULL,\
201 PRIMARY KEY (flow_id, id)"}
204 "flow_id INTEGER NOT NULL, norm_id INTEGER NOT NULL,\
205 side BOOLEAN NOT NULL, link_flow_id BOOLEAN NOT NULL,\
206 PRIMARY KEY (flow_id, norm_id, side)"}
209 "id INTEGER PRIMARY KEY AUTOINCREMENT,\
210 flow_id INTEGER NOT NULL, side INTEGER NOT NULL,\
211 amount DOUBLE NOT NULL, start INTEGER NOT NULL, paid INTEGER"}
214 "day INTEGER NOT NULL, id INTEGER NOT NULL,\
215 PRIMARY KEY (day, id)"}
218 "day INTEGER NOT NULL, event_id INTEGER\
219 NOT NULL, id INTEGER NOT NULL, amount DOUBLE NOT NULL,\
220 PRIMARY KEY (day, event_id, id)"}
222 /* XXX _acc This belongs to a separate module */
224 "day INTEGER NOT NULL, event_id INTEGER\
225 NOT NULL, transfer_id INTEGER NOT NULL, value DOUBLE NOT NULL,\
226 status INTEGER NOT NULL DEFAULT 0, earnings DOUBLE,\
227 PRIMARY KEY (day, event_id, transfer_id)"}
230 "resource_id INTEGER NOT NULL,\
231 day INTEGER NOT NULL, rate DOUBLE NOT NULL,\
232 diff DOUBLE NOT NULL, PRIMARY KEY (resource_id, day)"}
235 "id INTEGER PRIMARY KEY AUTOINCREMENT,\
236 flow_id INTEGER NOT NULL, side INTEGER NOT NULL,\
237 amount DOUBLE NOT NULL, value DOUBLE NOT NULL, start INTEGER NOT NULL,\
238 paid INTEGER, UNIQUE (flow_id, start)"}
241 "start INTEGER PRIMARY KEY NOT NULL,\
242 side INTEGER NOT NULL,\
243 value DOUBLE NOT NULL,\
247 "id INTEGER PRIMARY KEY NOT NULL,\
248 resource_id INTEGER NOT NULL"}
252 aaSession::getDatabase(nsISupports
*dbFile
)
256 nsCOMPtr
<nsIFile
> file
= do_QueryInterface(dbFile
);
259 nsCOMPtr
<aaIManager
> manager
=
260 do_CreateInstance(AA_MANAGER_CONTRACT
, &rv
);
261 NS_ENSURE_SUCCESS(rv
, rv
);
263 nsCOMPtr
<nsIFile
> file
;
264 rv
= manager
->TmpFileFromString("default.sqlite", PR_FALSE
,
265 getter_AddRefs(file
));
266 NS_ENSURE_SUCCESS(rv
, rv
);
269 nsCOMPtr
<mozIStorageService
> storageSvc
;
270 storageSvc
= do_GetService(MOZ_STORAGE_SERVICE_CONTRACTID
, &rv
);
271 NS_ENSURE_SUCCESS(rv
, rv
);
273 nsCOMPtr
<mozIStorageConnection
> connection
;
274 rv
= storageSvc
->OpenDatabase(file
, getter_AddRefs( connection
));
275 NS_ENSURE_SUCCESS(rv
, rv
);
278 for (i
= 0; i
< sizeof(tables
) / sizeof(struct aaTable
); i
++) {
279 rv
= checkTable(connection
, tables
[i
].name
, tables
[i
].sql
);
280 NS_ENSURE_SUCCESS(rv
, rv
);
282 mConnection
= connection
;
288 aaSession::checkTable(mozIStorageConnection
*aConnection
,
289 const char *tableName
, const char *createString
)
294 rv
= aConnection
->TableExists(nsDependentCString(tableName
), &hasTable
);
295 NS_ENSURE_SUCCESS(rv
, rv
);
297 rv
= aConnection
->CreateTable(tableName
, createString
);
298 NS_ENSURE_SUCCESS(rv
, rv
);