Branch data Line data Source code
1 : : /********************************************************************\
2 : : * Account.c -- Account data structure implementation *
3 : : * Copyright (C) 1997 Robin D. Clark *
4 : : * Copyright (C) 1997-2003 Linas Vepstas <linas@linas.org> *
5 : : * Copyright (C) 2007 David Hampton <hampton@employees.org> *
6 : : * *
7 : : * This program is free software; you can redistribute it and/or *
8 : : * modify it under the terms of the GNU General Public License as *
9 : : * published by the Free Software Foundation; either version 2 of *
10 : : * the License, or (at your option) any later version. *
11 : : * *
12 : : * This program is distributed in the hope that it will be useful, *
13 : : * but WITHOUT ANY WARRANTY; without even the implied warranty of *
14 : : * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
15 : : * GNU General Public License for more details. *
16 : : * *
17 : : * You should have received a copy of the GNU General Public License*
18 : : * along with this program; if not, contact: *
19 : : * *
20 : : * Free Software Foundation Voice: +1-617-542-5942 *
21 : : * 51 Franklin Street, Fifth Floor Fax: +1-617-542-2652 *
22 : : * Boston, MA 02110-1301, USA gnu@gnu.org *
23 : : * *
24 : : \********************************************************************/
25 : :
26 : : #include <config.h>
27 : :
28 : : #include "gnc-prefs.h"
29 : :
30 : : #include <glib.h>
31 : : #include <glib/gi18n.h>
32 : : #include <stdlib.h>
33 : : #include <stdint.h>
34 : : #include <string.h>
35 : :
36 : : #include "AccountP.hpp"
37 : : #include "Account.hpp"
38 : : #include "Split.h"
39 : : #include "Transaction.h"
40 : : #include "TransactionP.hpp"
41 : : #include "gnc-event.h"
42 : : #include "gnc-glib-utils.h"
43 : : #include "gnc-lot.h"
44 : : #include "gnc-pricedb.h"
45 : : #include "qofevent.h"
46 : : #include "qofinstance-p.h"
47 : : #include "gnc-features.h"
48 : : #include "guid.hpp"
49 : :
50 : : #include <numeric>
51 : : #include <map>
52 : : #include <unordered_set>
53 : : #include <algorithm>
54 : :
55 : : static QofLogModule log_module = GNC_MOD_ACCOUNT;
56 : :
57 : : /* The Canonical Account Separator. Pre-Initialized. */
58 : : static gchar account_separator[8] = ".";
59 : : static gunichar account_uc_separator = ':';
60 : :
61 : : static bool imap_convert_bayes_to_flat_run = false;
62 : :
63 : : /* Predefined KVP paths */
64 : : static const std::string KEY_ASSOC_INCOME_ACCOUNT("ofx/associated-income-account");
65 : : static const std::string KEY_RECONCILE_INFO("reconcile-info");
66 : : static const std::string KEY_INCLUDE_CHILDREN("include-children");
67 : : static const std::string KEY_POSTPONE("postpone");
68 : : static const std::string KEY_LOT_MGMT("lot-mgmt");
69 : : static const std::string KEY_ONLINE_ID("online_id");
70 : : static const std::string KEY_IMP_APPEND_TEXT("import-append-text");
71 : : static const std::string AB_KEY("hbci");
72 : : static const std::string AB_ACCOUNT_ID("account-id");
73 : : static const std::string AB_ACCOUNT_UID("account-uid");
74 : : static const std::string AB_BANK_CODE("bank-code");
75 : : static const std::string AB_TRANS_RETRIEVAL("trans-retrieval");
76 : :
77 : : static const std::string KEY_BALANCE_LIMIT("balance-limit");
78 : : static const std::string KEY_BALANCE_HIGHER_LIMIT_VALUE("higher-value");
79 : : static const std::string KEY_BALANCE_LOWER_LIMIT_VALUE("lower-value");
80 : : static const std::string KEY_BALANCE_INCLUDE_SUB_ACCTS("inlude-sub-accts");
81 : :
82 : : using FinalProbabilityVec=std::vector<std::pair<std::string, int32_t>>;
83 : : using ProbabilityVec=std::vector<std::pair<std::string, struct AccountProbability>>;
84 : : using FlatKvpEntry=std::pair<std::string, KvpValue*>;
85 : :
86 : : enum
87 : : {
88 : : LAST_SIGNAL
89 : : };
90 : :
91 : : enum
92 : : {
93 : : PROP_0,
94 : : PROP_NAME, /* Table */
95 : : PROP_FULL_NAME, /* Constructed */
96 : : PROP_CODE, /* Table */
97 : : PROP_DESCRIPTION, /* Table */
98 : : PROP_COLOR, /* KVP */
99 : : PROP_NOTES, /* KVP */
100 : : PROP_TYPE, /* Table */
101 : :
102 : : // PROP_PARENT, /* Table, Not a property */
103 : : PROP_COMMODITY, /* Table */
104 : : PROP_COMMODITY_SCU, /* Table */
105 : : PROP_NON_STD_SCU, /* Table */
106 : : PROP_END_BALANCE, /* Constructed */
107 : : PROP_END_NOCLOSING_BALANCE, /* Constructed */
108 : : PROP_END_CLEARED_BALANCE, /* Constructed */
109 : : PROP_END_RECONCILED_BALANCE, /* Constructed */
110 : :
111 : : PROP_TAX_RELATED, /* KVP */
112 : : PROP_TAX_CODE, /* KVP */
113 : : PROP_TAX_SOURCE, /* KVP */
114 : : PROP_TAX_COPY_NUMBER, /* KVP */
115 : :
116 : : PROP_HIDDEN, /* Table slot exists, but in KVP in memory & xml */
117 : : PROP_PLACEHOLDER, /* Table slot exists, but in KVP in memory & xml */
118 : : PROP_AUTO_INTEREST,
119 : : PROP_FILTER, /* KVP */
120 : : PROP_SORT_ORDER, /* KVP */
121 : : PROP_SORT_REVERSED,
122 : :
123 : : PROP_LOT_NEXT_ID, /* KVP */
124 : : PROP_ONLINE_ACCOUNT, /* KVP */
125 : : PROP_IMP_APPEND_TEXT, /* KVP */
126 : : PROP_IS_OPENING_BALANCE, /* KVP */
127 : : PROP_OFX_INCOME_ACCOUNT, /* KVP */
128 : : PROP_AB_ACCOUNT_ID, /* KVP */
129 : : PROP_AB_ACCOUNT_UID, /* KVP */
130 : : PROP_AB_BANK_CODE, /* KVP */
131 : : PROP_AB_TRANS_RETRIEVAL, /* KVP */
132 : :
133 : : PROP_RUNTIME_0,
134 : : PROP_POLICY, /* Cached Value */
135 : : PROP_MARK, /* Runtime Value */
136 : : PROP_SORT_DIRTY, /* Runtime Value */
137 : : PROP_BALANCE_DIRTY, /* Runtime Value */
138 : : PROP_START_BALANCE, /* Runtime Value */
139 : : PROP_START_NOCLOSING_BALANCE, /* Runtime Value */
140 : : PROP_START_CLEARED_BALANCE, /* Runtime Value */
141 : : PROP_START_RECONCILED_BALANCE, /* Runtime Value */
142 : : };
143 : :
144 : : #define GET_PRIVATE(o) \
145 : : ((AccountPrivate*)gnc_account_get_instance_private((Account*)o))
146 : :
147 : : /* This map contains a set of strings representing the different column types. */
148 : : static const std::map<GNCAccountType, const char*> gnc_acct_debit_strs = {
149 : : { ACCT_TYPE_NONE, N_("Funds In") },
150 : : { ACCT_TYPE_BANK, N_("Deposit") },
151 : : { ACCT_TYPE_CASH, N_("Receive") },
152 : : { ACCT_TYPE_CREDIT, N_("Payment") },
153 : : { ACCT_TYPE_ASSET, N_("Increase") },
154 : : { ACCT_TYPE_LIABILITY, N_("Decrease") },
155 : : { ACCT_TYPE_STOCK, N_("Buy") },
156 : : { ACCT_TYPE_MUTUAL, N_("Buy") },
157 : : { ACCT_TYPE_CURRENCY, N_("Buy") },
158 : : { ACCT_TYPE_INCOME, N_("Charge") },
159 : : { ACCT_TYPE_EXPENSE, N_("Expense") },
160 : : { ACCT_TYPE_PAYABLE, N_("Payment") },
161 : : { ACCT_TYPE_RECEIVABLE, N_("Invoice") },
162 : : { ACCT_TYPE_TRADING, N_("Decrease") },
163 : : { ACCT_TYPE_EQUITY, N_("Decrease") },
164 : : };
165 : : static const char* dflt_acct_debit_str = N_("Debit");
166 : :
167 : : /* This map contains a set of strings representing the different column types. */
168 : : static const std::map<GNCAccountType, const char*> gnc_acct_credit_strs = {
169 : : { ACCT_TYPE_NONE, N_("Funds Out") },
170 : : { ACCT_TYPE_BANK, N_("Withdrawal") },
171 : : { ACCT_TYPE_CASH, N_("Spend") },
172 : : { ACCT_TYPE_CREDIT, N_("Charge") },
173 : : { ACCT_TYPE_ASSET, N_("Decrease") },
174 : : { ACCT_TYPE_LIABILITY, N_("Increase") },
175 : : { ACCT_TYPE_STOCK, N_("Sell") },
176 : : { ACCT_TYPE_MUTUAL, N_("Sell") },
177 : : { ACCT_TYPE_CURRENCY, N_("Sell") },
178 : : { ACCT_TYPE_INCOME, N_("Income") },
179 : : { ACCT_TYPE_EXPENSE, N_("Rebate") },
180 : : { ACCT_TYPE_PAYABLE, N_("Bill") },
181 : : { ACCT_TYPE_RECEIVABLE, N_("Payment") },
182 : : { ACCT_TYPE_TRADING, N_("Increase") },
183 : : { ACCT_TYPE_EQUITY, N_("Increase") },
184 : : };
185 : : static const char* dflt_acct_credit_str = N_("Credit");
186 : :
187 : : /********************************************************************\
188 : : * Because I can't use C++ for this project, doesn't mean that I *
189 : : * can't pretend to! These functions perform actions on the *
190 : : * account data structure, in order to encapsulate the knowledge *
191 : : * of the internals of the Account in one file. *
192 : : \********************************************************************/
193 : :
194 : : static void xaccAccountBringUpToDate (Account *acc);
195 : :
196 : :
197 : : /********************************************************************\
198 : : * gnc_get_account_separator *
199 : : * returns the current account separator character *
200 : : * *
201 : : * Args: none *
202 : : * Returns: account separator character *
203 : : \*******************************************************************/
204 : : const gchar *
205 : 23 : gnc_get_account_separator_string (void)
206 : : {
207 : 23 : return account_separator;
208 : : }
209 : :
210 : : gunichar
211 : 3 : gnc_get_account_separator (void)
212 : : {
213 : 3 : return account_uc_separator;
214 : : }
215 : :
216 : : void
217 : 36 : gnc_set_account_separator (const gchar *separator)
218 : : {
219 : : gunichar uc;
220 : : gint count;
221 : :
222 : 36 : uc = g_utf8_get_char_validated(separator, -1);
223 : 36 : if ((uc == (gunichar) - 2) || (uc == (gunichar) - 1) || g_unichar_isalnum(uc))
224 : : {
225 : 1 : account_uc_separator = ':';
226 : 1 : strcpy(account_separator, ":");
227 : 1 : return;
228 : : }
229 : :
230 : 35 : account_uc_separator = uc;
231 : 35 : count = g_unichar_to_utf8(uc, account_separator);
232 : 35 : account_separator[count] = '\0';
233 : : }
234 : :
235 : 3 : gchar *gnc_account_name_violations_errmsg (const gchar *separator, GList* invalid_account_names)
236 : : {
237 : 3 : gchar *message = nullptr;
238 : :
239 : 3 : if ( !invalid_account_names )
240 : 2 : return nullptr;
241 : :
242 : 1 : auto account_list {gnc_g_list_stringjoin (invalid_account_names, "\n")};
243 : :
244 : : /* Translators: The first %s will be the account separator character,
245 : : the second %s is a list of account names.
246 : : The resulting string will be displayed to the user if there are
247 : : account names containing the separator character. */
248 : 1 : message = g_strdup_printf(
249 : 1 : _("The separator character \"%s\" is used in one or more account names.\n\n"
250 : : "This will result in unexpected behaviour. "
251 : : "Either change the account names or choose another separator character.\n\n"
252 : : "Below you will find the list of invalid account names:\n"
253 : : "%s"), separator, account_list );
254 : 1 : g_free ( account_list );
255 : 1 : return message;
256 : : }
257 : :
258 : : struct ViolationData
259 : : {
260 : : GList *list;
261 : : const gchar *separator;
262 : : };
263 : :
264 : : static void
265 : 3 : check_acct_name (Account *acct, gpointer user_data)
266 : : {
267 : 3 : auto cb {static_cast<ViolationData*>(user_data)};
268 : 3 : auto name {xaccAccountGetName (acct)};
269 : 3 : if (g_strstr_len (name, -1, cb->separator))
270 : 2 : cb->list = g_list_prepend (cb->list, g_strdup (name));
271 : 3 : }
272 : :
273 : 4 : GList *gnc_account_list_name_violations (QofBook *book, const gchar *separator)
274 : : {
275 : 4 : g_return_val_if_fail (separator != nullptr, nullptr);
276 : 2 : if (!book) return nullptr;
277 : 1 : ViolationData cb = { nullptr, separator };
278 : 1 : gnc_account_foreach_descendant (gnc_book_get_root_account (book),
279 : : (AccountCb)check_acct_name, &cb);
280 : 1 : return cb.list;
281 : : }
282 : :
283 : : /********************************************************************\
284 : : \********************************************************************/
285 : :
286 : : static inline void mark_account (Account *acc);
287 : : void
288 : 16697 : mark_account (Account *acc)
289 : : {
290 : 16697 : qof_instance_set_dirty(&acc->inst);
291 : 16697 : }
292 : :
293 : : /********************************************************************\
294 : : \********************************************************************/
295 : :
296 : : /* GObject Initialization */
297 : 1205198 : G_DEFINE_TYPE_WITH_PRIVATE(Account, gnc_account, QOF_TYPE_INSTANCE)
298 : :
299 : : static void
300 : 5047 : gnc_account_init(Account* acc)
301 : : {
302 : : AccountPrivate *priv;
303 : :
304 : 5047 : priv = GET_PRIVATE(acc);
305 : 5047 : priv->parent = nullptr;
306 : :
307 : 5047 : priv->accountName = qof_string_cache_insert("");
308 : 5047 : priv->accountCode = qof_string_cache_insert("");
309 : 5047 : priv->description = qof_string_cache_insert("");
310 : :
311 : 5047 : priv->type = ACCT_TYPE_NONE;
312 : :
313 : 5047 : priv->mark = 0;
314 : :
315 : 5047 : priv->policy = xaccGetFIFOPolicy();
316 : 5047 : priv->lots = nullptr;
317 : :
318 : 5047 : priv->commodity = nullptr;
319 : 5047 : priv->commodity_scu = 0;
320 : 5047 : priv->non_standard_scu = FALSE;
321 : :
322 : 5047 : priv->balance = gnc_numeric_zero();
323 : 5047 : priv->noclosing_balance = gnc_numeric_zero();
324 : 5047 : priv->cleared_balance = gnc_numeric_zero();
325 : 5047 : priv->reconciled_balance = gnc_numeric_zero();
326 : 5047 : priv->starting_balance = gnc_numeric_zero();
327 : 5047 : priv->starting_noclosing_balance = gnc_numeric_zero();
328 : 5047 : priv->starting_cleared_balance = gnc_numeric_zero();
329 : 5047 : priv->starting_reconciled_balance = gnc_numeric_zero();
330 : 5047 : priv->balance_dirty = FALSE;
331 : :
332 : 5047 : new (&priv->children) AccountVec ();
333 : 5047 : new (&priv->splits) SplitsVec ();
334 : 5047 : priv->splits_hash = g_hash_table_new (g_direct_hash, g_direct_equal);
335 : 5047 : priv->sort_dirty = FALSE;
336 : 5047 : }
337 : :
338 : : static void
339 : 2364 : gnc_account_dispose (GObject *acctp)
340 : : {
341 : 2364 : G_OBJECT_CLASS(gnc_account_parent_class)->dispose(acctp);
342 : 2364 : }
343 : :
344 : : static void
345 : 2364 : gnc_account_finalize(GObject* acctp)
346 : : {
347 : 2364 : G_OBJECT_CLASS(gnc_account_parent_class)->finalize(acctp);
348 : 2364 : }
349 : :
350 : : /* Note that g_value_set_object() refs the object, as does
351 : : * g_object_get(). But g_object_get() only unrefs once when it disgorges
352 : : * the object, leaving an unbalanced ref, which leaks. So instead of
353 : : * using g_value_set_object(), use g_value_take_object() which doesn't
354 : : * ref the object when used in get_property().
355 : : */
356 : : static void
357 : 597 : gnc_account_get_property (GObject *object,
358 : : guint prop_id,
359 : : GValue *value,
360 : : GParamSpec *pspec)
361 : : {
362 : : Account *account;
363 : : AccountPrivate *priv;
364 : :
365 : 597 : g_return_if_fail(GNC_IS_ACCOUNT(object));
366 : :
367 : 597 : account = GNC_ACCOUNT(object);
368 : 597 : priv = GET_PRIVATE(account);
369 : 597 : switch (prop_id)
370 : : {
371 : 96 : case PROP_NAME:
372 : 96 : g_value_set_string(value, priv->accountName);
373 : 96 : break;
374 : 1 : case PROP_FULL_NAME:
375 : 1 : g_value_take_string(value, gnc_account_get_full_name(account));
376 : 1 : break;
377 : 83 : case PROP_CODE:
378 : 83 : g_value_set_string(value, priv->accountCode);
379 : 83 : break;
380 : 75 : case PROP_DESCRIPTION:
381 : 75 : g_value_set_string(value, priv->description);
382 : 75 : break;
383 : 1 : case PROP_COLOR:
384 : 1 : g_value_set_string(value, xaccAccountGetColor(account));
385 : 1 : break;
386 : 1 : case PROP_NOTES:
387 : 1 : g_value_set_string(value, xaccAccountGetNotes(account));
388 : 1 : break;
389 : 35 : case PROP_TYPE:
390 : : // NEED TO BE CONVERTED TO A G_TYPE_ENUM
391 : 35 : g_value_set_int(value, priv->type);
392 : 35 : break;
393 : 41 : case PROP_COMMODITY:
394 : 41 : g_value_take_object(value, priv->commodity);
395 : 41 : break;
396 : 41 : case PROP_COMMODITY_SCU:
397 : 41 : g_value_set_int(value, priv->commodity_scu);
398 : 41 : break;
399 : 41 : case PROP_NON_STD_SCU:
400 : 41 : g_value_set_boolean(value, priv->non_standard_scu);
401 : 41 : break;
402 : 9 : case PROP_SORT_DIRTY:
403 : 9 : g_value_set_boolean(value, priv->sort_dirty);
404 : 9 : break;
405 : 9 : case PROP_BALANCE_DIRTY:
406 : 9 : g_value_set_boolean(value, priv->balance_dirty);
407 : 9 : break;
408 : 3 : case PROP_START_BALANCE:
409 : 3 : g_value_set_boxed(value, &priv->starting_balance);
410 : 3 : break;
411 : 0 : case PROP_START_NOCLOSING_BALANCE:
412 : 0 : g_value_set_boxed(value, &priv->starting_noclosing_balance);
413 : 0 : break;
414 : 1 : case PROP_START_CLEARED_BALANCE:
415 : 1 : g_value_set_boxed(value, &priv->starting_cleared_balance);
416 : 1 : break;
417 : 1 : case PROP_START_RECONCILED_BALANCE:
418 : 1 : g_value_set_boxed(value, &priv->starting_reconciled_balance);
419 : 1 : break;
420 : 3 : case PROP_END_BALANCE:
421 : 3 : g_value_set_boxed(value, &priv->balance);
422 : 3 : break;
423 : 0 : case PROP_END_NOCLOSING_BALANCE:
424 : 0 : g_value_set_boxed(value, &priv->noclosing_balance);
425 : 0 : break;
426 : 1 : case PROP_END_CLEARED_BALANCE:
427 : 1 : g_value_set_boxed(value, &priv->cleared_balance);
428 : 1 : break;
429 : 1 : case PROP_END_RECONCILED_BALANCE:
430 : 1 : g_value_set_boxed(value, &priv->reconciled_balance);
431 : 1 : break;
432 : 1 : case PROP_POLICY:
433 : : /* MAKE THIS A BOXED VALUE */
434 : 1 : g_value_set_pointer(value, priv->policy);
435 : 1 : break;
436 : 1 : case PROP_MARK:
437 : 1 : g_value_set_int(value, priv->mark);
438 : 1 : break;
439 : 1 : case PROP_TAX_RELATED:
440 : 1 : g_value_set_boolean(value, xaccAccountGetTaxRelated(account));
441 : 1 : break;
442 : 1 : case PROP_TAX_CODE:
443 : 1 : g_value_set_string(value, xaccAccountGetTaxUSCode(account));
444 : 1 : break;
445 : 1 : case PROP_TAX_SOURCE:
446 : 1 : g_value_set_string(value,
447 : : xaccAccountGetTaxUSPayerNameSource(account));
448 : 1 : break;
449 : 1 : case PROP_TAX_COPY_NUMBER:
450 : 1 : g_value_set_int64(value,
451 : : xaccAccountGetTaxUSCopyNumber(account));
452 : 1 : break;
453 : 41 : case PROP_HIDDEN:
454 : 41 : g_value_set_boolean(value, xaccAccountGetHidden(account));
455 : 41 : break;
456 : 0 : case PROP_AUTO_INTEREST:
457 : 0 : g_value_set_boolean (value, xaccAccountGetAutoInterest (account));
458 : 0 : break;
459 : 1 : case PROP_IS_OPENING_BALANCE:
460 : 1 : g_value_set_boolean(value, xaccAccountGetIsOpeningBalance(account));
461 : 1 : break;
462 : 41 : case PROP_PLACEHOLDER:
463 : 41 : g_value_set_boolean(value, xaccAccountGetPlaceholder(account));
464 : 41 : break;
465 : 0 : case PROP_FILTER:
466 : 0 : g_value_set_string(value, xaccAccountGetFilter(account));
467 : 0 : break;
468 : 0 : case PROP_SORT_ORDER:
469 : 0 : g_value_set_string(value, xaccAccountGetSortOrder(account));
470 : 0 : break;
471 : 0 : case PROP_SORT_REVERSED:
472 : 0 : g_value_set_boolean(value, xaccAccountGetSortReversed(account));
473 : 0 : break;
474 : 56 : case PROP_LOT_NEXT_ID:
475 : : /* Pre-set the value in case the frame is empty */
476 : 56 : g_value_set_int64 (value, 0);
477 : 224 : qof_instance_get_path_kvp (QOF_INSTANCE (account), value, {KEY_LOT_MGMT, "next-id"});
478 : 56 : break;
479 : 2 : case PROP_ONLINE_ACCOUNT:
480 : 6 : qof_instance_get_path_kvp (QOF_INSTANCE (account), value, {KEY_ONLINE_ID});
481 : 2 : break;
482 : 0 : case PROP_IMP_APPEND_TEXT:
483 : 0 : g_value_set_boolean(value, xaccAccountGetAppendText(account));
484 : 0 : break;
485 : 1 : case PROP_OFX_INCOME_ACCOUNT:
486 : 3 : qof_instance_get_path_kvp (QOF_INSTANCE (account), value, {KEY_ASSOC_INCOME_ACCOUNT});
487 : 1 : break;
488 : 1 : case PROP_AB_ACCOUNT_ID:
489 : 4 : qof_instance_get_path_kvp (QOF_INSTANCE (account), value, {AB_KEY, AB_ACCOUNT_ID});
490 : 1 : break;
491 : 1 : case PROP_AB_ACCOUNT_UID:
492 : 4 : qof_instance_get_path_kvp (QOF_INSTANCE (account), value, {AB_KEY, AB_ACCOUNT_UID});
493 : 1 : break;
494 : 1 : case PROP_AB_BANK_CODE:
495 : 4 : qof_instance_get_path_kvp (QOF_INSTANCE (account), value, {AB_KEY, AB_BANK_CODE});
496 : 1 : break;
497 : 3 : case PROP_AB_TRANS_RETRIEVAL:
498 : 12 : qof_instance_get_path_kvp (QOF_INSTANCE (account), value, {AB_KEY, AB_TRANS_RETRIEVAL});
499 : 3 : break;
500 : 0 : default:
501 : 0 : G_OBJECT_WARN_INVALID_PROPERTY_ID(object, prop_id, pspec);
502 : 0 : break;
503 : : }
504 : 177 : }
505 : :
506 : : static void
507 : 70750 : gnc_account_set_property (GObject *object,
508 : : guint prop_id,
509 : : const GValue *value,
510 : : GParamSpec *pspec)
511 : : {
512 : : Account *account;
513 : : gnc_numeric *number;
514 : 70750 : g_return_if_fail(GNC_IS_ACCOUNT(object));
515 : 70750 : account = GNC_ACCOUNT(object);
516 : 70750 : if (prop_id < PROP_RUNTIME_0)
517 : 483 : g_assert (qof_instance_get_editlevel(account));
518 : :
519 : 70750 : switch (prop_id)
520 : : {
521 : 50 : case PROP_NAME:
522 : 50 : xaccAccountSetName(account, g_value_get_string(value));
523 : 50 : break;
524 : 46 : case PROP_CODE:
525 : 46 : xaccAccountSetCode(account, g_value_get_string(value));
526 : 46 : break;
527 : 42 : case PROP_DESCRIPTION:
528 : 42 : xaccAccountSetDescription(account, g_value_get_string(value));
529 : 42 : break;
530 : 0 : case PROP_COLOR:
531 : 0 : xaccAccountSetColor(account, g_value_get_string(value));
532 : 0 : break;
533 : 0 : case PROP_NOTES:
534 : 0 : xaccAccountSetNotes(account, g_value_get_string(value));
535 : 0 : break;
536 : 21 : case PROP_TYPE:
537 : : // NEED TO BE CONVERTED TO A G_TYPE_ENUM
538 : 21 : xaccAccountSetType(account, static_cast<GNCAccountType>(g_value_get_int(value)));
539 : 21 : break;
540 : 29 : case PROP_COMMODITY:
541 : 29 : xaccAccountSetCommodity(account, static_cast<gnc_commodity*>(g_value_get_object(value)));
542 : 29 : break;
543 : 42 : case PROP_COMMODITY_SCU:
544 : 42 : xaccAccountSetCommoditySCU(account, g_value_get_int(value));
545 : 42 : break;
546 : 42 : case PROP_NON_STD_SCU:
547 : 42 : xaccAccountSetNonStdSCU(account, g_value_get_boolean(value));
548 : 42 : break;
549 : 35133 : case PROP_SORT_DIRTY:
550 : 35133 : gnc_account_set_sort_dirty(account);
551 : 35133 : break;
552 : 35133 : case PROP_BALANCE_DIRTY:
553 : 35133 : gnc_account_set_balance_dirty(account);
554 : 35133 : break;
555 : 1 : case PROP_START_BALANCE:
556 : 1 : number = static_cast<gnc_numeric*>(g_value_get_boxed(value));
557 : 1 : gnc_account_set_start_balance(account, *number);
558 : 1 : break;
559 : 0 : case PROP_START_CLEARED_BALANCE:
560 : 0 : number = static_cast<gnc_numeric*>(g_value_get_boxed(value));
561 : 0 : gnc_account_set_start_cleared_balance(account, *number);
562 : 0 : break;
563 : 0 : case PROP_START_RECONCILED_BALANCE:
564 : 0 : number = static_cast<gnc_numeric*>(g_value_get_boxed(value));
565 : 0 : gnc_account_set_start_reconciled_balance(account, *number);
566 : 0 : break;
567 : 0 : case PROP_POLICY:
568 : 0 : gnc_account_set_policy(account, static_cast<GNCPolicy*>(g_value_get_pointer(value)));
569 : 0 : break;
570 : 0 : case PROP_MARK:
571 : 0 : xaccAccountSetMark(account, g_value_get_int(value));
572 : 0 : break;
573 : 0 : case PROP_TAX_RELATED:
574 : 0 : xaccAccountSetTaxRelated(account, g_value_get_boolean(value));
575 : 0 : break;
576 : 0 : case PROP_TAX_CODE:
577 : 0 : xaccAccountSetTaxUSCode(account, g_value_get_string(value));
578 : 0 : break;
579 : 0 : case PROP_TAX_SOURCE:
580 : 0 : xaccAccountSetTaxUSPayerNameSource(account,
581 : : g_value_get_string(value));
582 : 0 : break;
583 : 0 : case PROP_TAX_COPY_NUMBER:
584 : 0 : xaccAccountSetTaxUSCopyNumber(account,
585 : : g_value_get_int64(value));
586 : 0 : break;
587 : 42 : case PROP_HIDDEN:
588 : 42 : xaccAccountSetHidden(account, g_value_get_boolean(value));
589 : 42 : break;
590 : 0 : case PROP_AUTO_INTEREST:
591 : 0 : xaccAccountSetAutoInterest (account, g_value_get_boolean (value));
592 : 0 : break;
593 : 0 : case PROP_IS_OPENING_BALANCE:
594 : 0 : xaccAccountSetIsOpeningBalance (account, g_value_get_boolean (value));
595 : 0 : break;
596 : 42 : case PROP_PLACEHOLDER:
597 : 42 : xaccAccountSetPlaceholder(account, g_value_get_boolean(value));
598 : 42 : break;
599 : 0 : case PROP_FILTER:
600 : 0 : xaccAccountSetFilter(account, g_value_get_string(value));
601 : 0 : break;
602 : 0 : case PROP_SORT_ORDER:
603 : 0 : xaccAccountSetSortOrder(account, g_value_get_string(value));
604 : 0 : break;
605 : 0 : case PROP_SORT_REVERSED:
606 : 0 : xaccAccountSetSortReversed(account, g_value_get_boolean(value));
607 : 0 : break;
608 : 56 : case PROP_LOT_NEXT_ID:
609 : 224 : qof_instance_set_path_kvp (QOF_INSTANCE (account), value, {KEY_LOT_MGMT, "next-id"});
610 : 56 : break;
611 : 64 : case PROP_ONLINE_ACCOUNT:
612 : 192 : qof_instance_set_path_kvp (QOF_INSTANCE (account), value, {KEY_ONLINE_ID});
613 : 64 : break;
614 : 0 : case PROP_IMP_APPEND_TEXT:
615 : 0 : xaccAccountSetAppendText(account, g_value_get_boolean(value));
616 : 0 : break;
617 : 2 : case PROP_OFX_INCOME_ACCOUNT:
618 : 6 : qof_instance_set_path_kvp (QOF_INSTANCE (account), value, {KEY_ASSOC_INCOME_ACCOUNT});
619 : 2 : break;
620 : 1 : case PROP_AB_ACCOUNT_ID:
621 : 4 : qof_instance_set_path_kvp (QOF_INSTANCE (account), value, {AB_KEY, AB_ACCOUNT_ID});
622 : 1 : break;
623 : 1 : case PROP_AB_ACCOUNT_UID:
624 : 4 : qof_instance_set_path_kvp (QOF_INSTANCE (account), value, {AB_KEY, AB_ACCOUNT_UID});
625 : 1 : break;
626 : 1 : case PROP_AB_BANK_CODE:
627 : 4 : qof_instance_set_path_kvp (QOF_INSTANCE (account), value, {AB_KEY, AB_BANK_CODE});
628 : 1 : break;
629 : 2 : case PROP_AB_TRANS_RETRIEVAL:
630 : 8 : qof_instance_set_path_kvp (QOF_INSTANCE (account), value, {AB_KEY, AB_TRANS_RETRIEVAL});
631 : 2 : break;
632 : 0 : default:
633 : 0 : G_OBJECT_WARN_INVALID_PROPERTY_ID(object, prop_id, pspec);
634 : 0 : break;
635 : : }
636 : 239 : }
637 : :
638 : : static void
639 : 67 : gnc_account_class_init (AccountClass *klass)
640 : : {
641 : 67 : GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
642 : :
643 : 67 : gobject_class->dispose = gnc_account_dispose;
644 : 67 : gobject_class->finalize = gnc_account_finalize;
645 : 67 : gobject_class->set_property = gnc_account_set_property;
646 : 67 : gobject_class->get_property = gnc_account_get_property;
647 : :
648 : : g_object_class_install_property
649 : 67 : (gobject_class,
650 : : PROP_NAME,
651 : : g_param_spec_string ("name",
652 : : "Account Name",
653 : : "The accountName is an arbitrary string "
654 : : "assigned by the user. It is intended to "
655 : : "a short, 5 to 30 character long string "
656 : : "that is displayed by the GUI as the "
657 : : "account mnemonic. Account names may be "
658 : : "repeated. but no two accounts that share "
659 : : "a parent may have the same name.",
660 : : nullptr,
661 : : static_cast<GParamFlags>(G_PARAM_READWRITE)));
662 : :
663 : : g_object_class_install_property
664 : 67 : (gobject_class,
665 : : PROP_FULL_NAME,
666 : : g_param_spec_string ("fullname",
667 : : "Full Account Name",
668 : : "The name of the account concatenated with "
669 : : "all its parent account names to indicate "
670 : : "a unique account.",
671 : : nullptr,
672 : : static_cast<GParamFlags>(G_PARAM_READABLE)));
673 : :
674 : : g_object_class_install_property
675 : 67 : (gobject_class,
676 : : PROP_CODE,
677 : : g_param_spec_string ("code",
678 : : "Account Code",
679 : : "The account code is an arbitrary string "
680 : : "assigned by the user. It is intended to "
681 : : "be reporting code that is a synonym for "
682 : : "the accountName.",
683 : : nullptr,
684 : : static_cast<GParamFlags>(G_PARAM_READWRITE)));
685 : :
686 : : g_object_class_install_property
687 : 67 : (gobject_class,
688 : : PROP_DESCRIPTION,
689 : : g_param_spec_string ("description",
690 : : "Account Description",
691 : : "The account description is an arbitrary "
692 : : "string assigned by the user. It is intended "
693 : : "to be a longer, 1-5 sentence description of "
694 : : "what this account is all about.",
695 : : nullptr,
696 : : static_cast<GParamFlags>(G_PARAM_READWRITE)));
697 : :
698 : : g_object_class_install_property
699 : 67 : (gobject_class,
700 : : PROP_COLOR,
701 : : g_param_spec_string ("color",
702 : : "Account Color",
703 : : "The account color is a color string assigned "
704 : : "by the user. It is intended to highlight the "
705 : : "account based on the users wishes.",
706 : : nullptr,
707 : : static_cast<GParamFlags>(G_PARAM_READWRITE)));
708 : :
709 : : g_object_class_install_property
710 : 67 : (gobject_class,
711 : : PROP_NOTES,
712 : : g_param_spec_string ("notes",
713 : : "Account Notes",
714 : : "The account notes is an arbitrary provided "
715 : : "for the user to attach any other text that "
716 : : "they would like to associate with the account.",
717 : : nullptr,
718 : : static_cast<GParamFlags>(G_PARAM_READWRITE)));
719 : :
720 : : g_object_class_install_property
721 : 67 : (gobject_class,
722 : : PROP_TYPE,
723 : : g_param_spec_int ("type",
724 : : "Account Type",
725 : : "The account type, picked from the enumerated list "
726 : : "that includes ACCT_TYPE_BANK, ACCT_TYPE_STOCK, "
727 : : "ACCT_TYPE_CREDIT, ACCT_TYPE_INCOME, etc.",
728 : : ACCT_TYPE_NONE,
729 : : NUM_ACCOUNT_TYPES - 1,
730 : : ACCT_TYPE_BANK,
731 : : static_cast<GParamFlags>(G_PARAM_READWRITE)));
732 : :
733 : : g_object_class_install_property
734 : 67 : (gobject_class,
735 : : PROP_COMMODITY,
736 : : g_param_spec_object ("commodity",
737 : : "Commodity",
738 : : "The commodity field denotes the kind of "
739 : : "'stuff' stored in this account, whether "
740 : : "it is USD, gold, stock, etc.",
741 : : GNC_TYPE_COMMODITY,
742 : : static_cast<GParamFlags>(G_PARAM_READWRITE)));
743 : :
744 : : g_object_class_install_property
745 : 67 : (gobject_class,
746 : : PROP_COMMODITY_SCU,
747 : : g_param_spec_int ("commodity-scu",
748 : : "Commodity SCU",
749 : : "The smallest fraction of the commodity that is "
750 : : "tracked. This number is used as the denominator "
751 : : "value in 1/x, so a value of 100 says that the "
752 : : "commodity can be divided into hundredths. E.G."
753 : : "1 USD can be divided into 100 cents.",
754 : : 0,
755 : : G_MAXINT32,
756 : : GNC_COMMODITY_MAX_FRACTION,
757 : : static_cast<GParamFlags>(G_PARAM_READWRITE)));
758 : :
759 : : g_object_class_install_property
760 : 67 : (gobject_class,
761 : : PROP_NON_STD_SCU,
762 : : g_param_spec_boolean ("non-std-scu",
763 : : "Non-std SCU",
764 : : "TRUE if the account SCU doesn't match "
765 : : "the commodity SCU. This indicates a case "
766 : : "where the two were accidentally set to "
767 : : "mismatched values in older versions of "
768 : : "GnuCash.",
769 : : FALSE,
770 : : static_cast<GParamFlags>(G_PARAM_READWRITE)));
771 : :
772 : : g_object_class_install_property
773 : 67 : (gobject_class,
774 : : PROP_SORT_DIRTY,
775 : : g_param_spec_boolean("sort-dirty",
776 : : "Sort Dirty",
777 : : "TRUE if the splits in the account needs to be "
778 : : "resorted. This flag is set by the accounts "
779 : : "code for certain internal modifications, or "
780 : : "when external code calls the engine to say a "
781 : : "split has been modified in a way that may "
782 : : "affect the sort order of the account. Note: "
783 : : "This value can only be set to TRUE.",
784 : : FALSE,
785 : : static_cast<GParamFlags>(G_PARAM_READWRITE)));
786 : :
787 : : g_object_class_install_property
788 : 67 : (gobject_class,
789 : : PROP_BALANCE_DIRTY,
790 : : g_param_spec_boolean("balance-dirty",
791 : : "Balance Dirty",
792 : : "TRUE if the running balances in the account "
793 : : "needs to be recalculated. This flag is set "
794 : : "by the accounts code for certain internal "
795 : : "modifications, or when external code calls "
796 : : "the engine to say a split has been modified. "
797 : : "Note: This value can only be set to TRUE.",
798 : : FALSE,
799 : : static_cast<GParamFlags>(G_PARAM_READWRITE)));
800 : :
801 : : g_object_class_install_property
802 : 67 : (gobject_class,
803 : : PROP_START_BALANCE,
804 : : g_param_spec_boxed("start-balance",
805 : : "Starting Balance",
806 : : "The starting balance for the account. This "
807 : : "parameter is intended for use with backends that "
808 : : "do not return the complete list of splits for an "
809 : : "account, but rather return a partial list. In "
810 : : "such a case, the backend will typically return "
811 : : "all of the splits after some certain date, and "
812 : : "the 'starting balance' will represent the "
813 : : "summation of the splits up to that date.",
814 : : GNC_TYPE_NUMERIC,
815 : : static_cast<GParamFlags>(G_PARAM_READWRITE)));
816 : :
817 : : g_object_class_install_property
818 : 67 : (gobject_class,
819 : : PROP_START_NOCLOSING_BALANCE,
820 : : g_param_spec_boxed("start-noclosing-balance",
821 : : "Starting No-closing Balance",
822 : : "The starting balance for the account, ignoring closing."
823 : : "This parameter is intended for use with backends "
824 : : "that do not return the complete list of splits "
825 : : "for an account, but rather return a partial "
826 : : "list. In such a case, the backend will "
827 : : "typically return all of the splits after "
828 : : "some certain date, and the 'starting noclosing "
829 : : "balance' will represent the summation of the "
830 : : "splits up to that date, ignoring closing splits.",
831 : : GNC_TYPE_NUMERIC,
832 : : static_cast<GParamFlags>(G_PARAM_READWRITE)));
833 : :
834 : : g_object_class_install_property
835 : 67 : (gobject_class,
836 : : PROP_START_CLEARED_BALANCE,
837 : : g_param_spec_boxed("start-cleared-balance",
838 : : "Starting Cleared Balance",
839 : : "The starting cleared balance for the account. "
840 : : "This parameter is intended for use with backends "
841 : : "that do not return the complete list of splits "
842 : : "for an account, but rather return a partial "
843 : : "list. In such a case, the backend will "
844 : : "typically return all of the splits after "
845 : : "some certain date, and the 'starting cleared "
846 : : "balance' will represent the summation of the "
847 : : "splits up to that date.",
848 : : GNC_TYPE_NUMERIC,
849 : : static_cast<GParamFlags>(G_PARAM_READWRITE)));
850 : :
851 : : g_object_class_install_property
852 : 67 : (gobject_class,
853 : : PROP_START_RECONCILED_BALANCE,
854 : : g_param_spec_boxed("start-reconciled-balance",
855 : : "Starting Reconciled Balance",
856 : : "The starting reconciled balance for the "
857 : : "account. This parameter is intended for use "
858 : : "with backends that do not return the complete "
859 : : "list of splits for an account, but rather return "
860 : : "a partial list. In such a case, the backend "
861 : : "will typically return all of the splits after "
862 : : "some certain date, and the 'starting reconciled "
863 : : "balance' will represent the summation of the "
864 : : "splits up to that date.",
865 : : GNC_TYPE_NUMERIC,
866 : : static_cast<GParamFlags>(G_PARAM_READWRITE)));
867 : :
868 : : g_object_class_install_property
869 : 67 : (gobject_class,
870 : : PROP_END_BALANCE,
871 : : g_param_spec_boxed("end-balance",
872 : : "Ending Account Balance",
873 : : "This is the current ending balance for the "
874 : : "account. It is computed from the sum of the "
875 : : "starting balance and all splits in the account.",
876 : : GNC_TYPE_NUMERIC,
877 : : G_PARAM_READABLE));
878 : :
879 : : g_object_class_install_property
880 : 67 : (gobject_class,
881 : : PROP_END_NOCLOSING_BALANCE,
882 : : g_param_spec_boxed("end-noclosing-balance",
883 : : "Ending Account Noclosing Balance",
884 : : "This is the current ending no-closing balance for "
885 : : "the account. It is computed from the sum of the "
886 : : "starting balance and all cleared splits in the "
887 : : "account.",
888 : : GNC_TYPE_NUMERIC,
889 : : G_PARAM_READABLE));
890 : :
891 : : g_object_class_install_property
892 : 67 : (gobject_class,
893 : : PROP_END_CLEARED_BALANCE,
894 : : g_param_spec_boxed("end-cleared-balance",
895 : : "Ending Account Cleared Balance",
896 : : "This is the current ending cleared balance for "
897 : : "the account. It is computed from the sum of the "
898 : : "starting balance and all cleared splits in the "
899 : : "account.",
900 : : GNC_TYPE_NUMERIC,
901 : : G_PARAM_READABLE));
902 : :
903 : : g_object_class_install_property
904 : 67 : (gobject_class,
905 : : PROP_END_RECONCILED_BALANCE,
906 : : g_param_spec_boxed("end-reconciled-balance",
907 : : "Ending Account Reconciled Balance",
908 : : "This is the current ending reconciled balance "
909 : : "for the account. It is computed from the sum of "
910 : : "the starting balance and all reconciled splits "
911 : : "in the account.",
912 : : GNC_TYPE_NUMERIC,
913 : : static_cast<GParamFlags>(G_PARAM_READABLE)));
914 : :
915 : : g_object_class_install_property
916 : 67 : (gobject_class,
917 : : PROP_POLICY,
918 : : g_param_spec_pointer ("policy",
919 : : "Policy",
920 : : "The account lots policy.",
921 : : static_cast<GParamFlags>(G_PARAM_READWRITE)));
922 : :
923 : : g_object_class_install_property
924 : 67 : (gobject_class,
925 : : PROP_MARK,
926 : : g_param_spec_int ("acct-mark",
927 : : "Account Mark",
928 : : "Ipsum Lorem",
929 : : 0,
930 : : G_MAXINT16,
931 : : 0,
932 : : static_cast<GParamFlags>(G_PARAM_READWRITE)));
933 : :
934 : : g_object_class_install_property
935 : 67 : (gobject_class,
936 : : PROP_TAX_RELATED,
937 : : g_param_spec_boolean ("tax-related",
938 : : "Tax Related",
939 : : "Whether the account maps to an entry on an "
940 : : "income tax document.",
941 : : FALSE,
942 : : static_cast<GParamFlags>(G_PARAM_READWRITE)));
943 : :
944 : : g_object_class_install_property
945 : 67 : (gobject_class,
946 : : PROP_IS_OPENING_BALANCE,
947 : : g_param_spec_boolean ("opening-balance",
948 : : "Opening Balance",
949 : : "Whether the account holds opening balances",
950 : : FALSE,
951 : : static_cast<GParamFlags>(G_PARAM_READWRITE)));
952 : :
953 : : g_object_class_install_property
954 : 67 : (gobject_class,
955 : : PROP_TAX_CODE,
956 : : g_param_spec_string ("tax-code",
957 : : "Tax Code",
958 : : "This is the code for mapping an account to a "
959 : : "specific entry on a taxable document. In the "
960 : : "United States it is used to transfer totals "
961 : : "into tax preparation software.",
962 : : nullptr,
963 : : static_cast<GParamFlags>(G_PARAM_READWRITE)));
964 : :
965 : : g_object_class_install_property
966 : 67 : (gobject_class,
967 : : PROP_TAX_SOURCE,
968 : : g_param_spec_string ("tax-source",
969 : : "Tax Source",
970 : : "This specifies where exported name comes from.",
971 : : nullptr,
972 : : static_cast<GParamFlags>(G_PARAM_READWRITE)));
973 : :
974 : : g_object_class_install_property
975 : 67 : (gobject_class,
976 : : PROP_TAX_COPY_NUMBER,
977 : : g_param_spec_int64 ("tax-copy-number",
978 : : "Tax Copy Number",
979 : : "This specifies the copy number of the tax "
980 : : "form/schedule.",
981 : : (gint64)1,
982 : : G_MAXINT64,
983 : : (gint64)1,
984 : : static_cast<GParamFlags>(G_PARAM_READWRITE)));
985 : :
986 : : g_object_class_install_property
987 : 67 : (gobject_class,
988 : : PROP_HIDDEN,
989 : : g_param_spec_boolean ("hidden",
990 : : "Hidden",
991 : : "Whether the account should be hidden in the "
992 : : "account tree.",
993 : : FALSE,
994 : : static_cast<GParamFlags>(G_PARAM_READWRITE)));
995 : :
996 : : g_object_class_install_property
997 : 67 : (gobject_class,
998 : : PROP_AUTO_INTEREST,
999 : : g_param_spec_boolean ("auto-interest-transfer",
1000 : : "Auto Interest",
1001 : : "Whether an interest transfer should be automatically "
1002 : : "added before reconcile.",
1003 : : FALSE,
1004 : : static_cast<GParamFlags>(G_PARAM_READWRITE)));
1005 : :
1006 : : g_object_class_install_property
1007 : 67 : (gobject_class,
1008 : : PROP_PLACEHOLDER,
1009 : : g_param_spec_boolean ("placeholder",
1010 : : "Placeholder",
1011 : : "Whether the account is a placeholder account which does not "
1012 : : "allow transactions to be created, edited or deleted.",
1013 : : FALSE,
1014 : : static_cast<GParamFlags>(G_PARAM_READWRITE)));
1015 : :
1016 : : g_object_class_install_property
1017 : 67 : (gobject_class,
1018 : : PROP_FILTER,
1019 : : g_param_spec_string ("filter",
1020 : : "Account Filter",
1021 : : "The account filter is a value saved to allow "
1022 : : "filters to be recalled.",
1023 : : nullptr,
1024 : : static_cast<GParamFlags>(G_PARAM_READWRITE)));
1025 : :
1026 : : g_object_class_install_property
1027 : 67 : (gobject_class,
1028 : : PROP_SORT_ORDER,
1029 : : g_param_spec_string ("sort-order",
1030 : : "Account Sort Order",
1031 : : "The account sort order is a value saved to allow "
1032 : : "the sort order to be recalled.",
1033 : : nullptr,
1034 : : static_cast<GParamFlags>(G_PARAM_READWRITE)));
1035 : :
1036 : : g_object_class_install_property
1037 : 67 : (gobject_class,
1038 : : PROP_SORT_REVERSED,
1039 : : g_param_spec_boolean ("sort-reversed",
1040 : : "Account Sort Reversed",
1041 : : "Parameter to store whether the sort order is reversed or not.",
1042 : : FALSE,
1043 : : static_cast<GParamFlags>(G_PARAM_READWRITE)));
1044 : :
1045 : : g_object_class_install_property
1046 : 67 : (gobject_class,
1047 : : PROP_LOT_NEXT_ID,
1048 : : g_param_spec_int64 ("lot-next-id",
1049 : : "Lot Next ID",
1050 : : "Tracks the next id to use in gnc_lot_make_default.",
1051 : : (gint64)1,
1052 : : G_MAXINT64,
1053 : : (gint64)1,
1054 : : static_cast<GParamFlags>(G_PARAM_READWRITE)));
1055 : :
1056 : : g_object_class_install_property
1057 : 67 : (gobject_class,
1058 : : PROP_ONLINE_ACCOUNT,
1059 : : g_param_spec_string ("online-id",
1060 : : "Online Account ID",
1061 : : "The online account which corresponds to this "
1062 : : "account for OFX import",
1063 : : nullptr,
1064 : : static_cast<GParamFlags>(G_PARAM_READWRITE)));
1065 : :
1066 : : g_object_class_install_property
1067 : 67 : (gobject_class,
1068 : : PROP_IMP_APPEND_TEXT,
1069 : : g_param_spec_boolean ("import-append-text",
1070 : : "Import Append Text",
1071 : : "Saved state of Append checkbox for setting initial "
1072 : : "value next time this account is imported.",
1073 : : FALSE,
1074 : : static_cast<GParamFlags>(G_PARAM_READWRITE)));
1075 : :
1076 : 67 : g_object_class_install_property(
1077 : : gobject_class,
1078 : : PROP_OFX_INCOME_ACCOUNT,
1079 : : g_param_spec_boxed("ofx-income-account",
1080 : : "Associated income account",
1081 : : "Used by the OFX importer.",
1082 : : GNC_TYPE_GUID,
1083 : : static_cast<GParamFlags>(G_PARAM_READWRITE)));
1084 : :
1085 : : g_object_class_install_property
1086 : 67 : (gobject_class,
1087 : : PROP_AB_ACCOUNT_ID,
1088 : : g_param_spec_string ("ab-account-id",
1089 : : "AQBanking Account ID",
1090 : : "The AqBanking account which corresponds to this "
1091 : : "account for AQBanking import",
1092 : : nullptr,
1093 : : static_cast<GParamFlags>(G_PARAM_READWRITE)));
1094 : : g_object_class_install_property
1095 : 67 : (gobject_class,
1096 : : PROP_AB_BANK_CODE,
1097 : : g_param_spec_string ("ab-bank-code",
1098 : : "AQBanking Bank Code",
1099 : : "The online account which corresponds to this "
1100 : : "account for AQBanking import",
1101 : : nullptr,
1102 : : static_cast<GParamFlags>(G_PARAM_READWRITE)));
1103 : :
1104 : : g_object_class_install_property
1105 : 67 : (gobject_class,
1106 : : PROP_AB_ACCOUNT_UID,
1107 : : g_param_spec_int64 ("ab-account-uid",
1108 : : "AQBanking Account UID",
1109 : : "Tracks the next id to use in gnc_lot_make_default.",
1110 : : (gint64)1,
1111 : : G_MAXINT64,
1112 : : (gint64)1,
1113 : : static_cast<GParamFlags>(G_PARAM_READWRITE)));
1114 : :
1115 : : g_object_class_install_property
1116 : 67 : (gobject_class,
1117 : : PROP_AB_TRANS_RETRIEVAL,
1118 : : g_param_spec_boxed("ab-trans-retrieval",
1119 : : "AQBanking Last Transaction Retrieval",
1120 : : "The time of the last transaction retrieval for this "
1121 : : "account.",
1122 : : GNC_TYPE_TIME64,
1123 : : static_cast<GParamFlags>(G_PARAM_READWRITE)));
1124 : :
1125 : 67 : }
1126 : :
1127 : : static void
1128 : 5045 : xaccInitAccount (Account * acc, QofBook *book)
1129 : : {
1130 : 5045 : ENTER ("book=%p\n", book);
1131 : 5045 : qof_instance_init_data (&acc->inst, GNC_ID_ACCOUNT, book);
1132 : :
1133 : 5045 : LEAVE ("account=%p\n", acc);
1134 : 5045 : }
1135 : :
1136 : : /********************************************************************\
1137 : : \********************************************************************/
1138 : :
1139 : : void
1140 : 1397 : gnc_account_foreach_split (const Account *acc, std::function<void(Split*)> func)
1141 : : {
1142 : 1397 : if (!GNC_IS_ACCOUNT (acc))
1143 : 0 : return;
1144 : :
1145 : 1397 : auto& splits{GET_PRIVATE(acc)->splits};
1146 : 1397 : std::for_each (splits.begin(), splits.end(), func);
1147 : : }
1148 : :
1149 : : void
1150 : 7124 : gnc_account_foreach_split_until_date (const Account *acc, time64 end_date,
1151 : : std::function<void(Split*)> f)
1152 : : {
1153 : 7124 : if (!GNC_IS_ACCOUNT (acc))
1154 : 0 : return;
1155 : :
1156 : 8574 : auto after_date = [](time64 end_date, auto s) -> bool
1157 : 8574 : { return (xaccTransGetDate (xaccSplitGetParent (s)) > end_date); };
1158 : :
1159 : 7124 : auto& splits{GET_PRIVATE(acc)->splits};
1160 : 7124 : auto after_date_iter = std::upper_bound (splits.begin(), splits.end(), end_date, after_date);
1161 : 7124 : std::for_each (splits.begin(), after_date_iter, f);
1162 : : }
1163 : :
1164 : :
1165 : : Split*
1166 : 2986 : gnc_account_find_split (const Account *acc, std::function<bool(const Split*)> predicate,
1167 : : bool reverse)
1168 : : {
1169 : 2986 : if (!GNC_IS_ACCOUNT (acc))
1170 : 0 : return nullptr;
1171 : :
1172 : 2986 : const auto& splits{GET_PRIVATE(acc)->splits};
1173 : 2986 : if (reverse)
1174 : : {
1175 : 2938 : auto latest = std::find_if(splits.rbegin(), splits.rend(), predicate);
1176 : 2938 : return (latest == splits.rend()) ? nullptr : *latest;
1177 : : }
1178 : : else
1179 : : {
1180 : 48 : auto earliest = std::find_if(splits.begin(), splits.end(), predicate);
1181 : 48 : return (earliest == splits.end()) ? nullptr : *earliest;
1182 : : }
1183 : : }
1184 : :
1185 : : /********************************************************************\
1186 : : \********************************************************************/
1187 : :
1188 : : QofBook *
1189 : 1135 : gnc_account_get_book(const Account *account)
1190 : : {
1191 : 1135 : if (!account) return nullptr;
1192 : 1135 : return qof_instance_get_book(QOF_INSTANCE(account));
1193 : : }
1194 : :
1195 : : /********************************************************************\
1196 : : \********************************************************************/
1197 : :
1198 : : static Account *
1199 : 2818 : gnc_coll_get_root_account (QofCollection *col)
1200 : : {
1201 : 2818 : if (!col) return nullptr;
1202 : 2818 : return static_cast<Account*>(qof_collection_get_data (col));
1203 : : }
1204 : :
1205 : : static void
1206 : 331 : gnc_coll_set_root_account (QofCollection *col, Account *root)
1207 : : {
1208 : : AccountPrivate *rpriv;
1209 : : Account *old_root;
1210 : 331 : if (!col) return;
1211 : :
1212 : 331 : old_root = gnc_coll_get_root_account (col);
1213 : 331 : if (old_root == root) return;
1214 : :
1215 : : /* If the new root is already linked into the tree somewhere, then
1216 : : * remove it from its current position before adding it at the
1217 : : * top. */
1218 : 330 : rpriv = GET_PRIVATE(root);
1219 : 330 : if (rpriv->parent)
1220 : : {
1221 : 20 : xaccAccountBeginEdit(root);
1222 : 20 : gnc_account_remove_child(rpriv->parent, root);
1223 : 20 : xaccAccountCommitEdit(root);
1224 : : }
1225 : :
1226 : 330 : qof_collection_set_data (col, root);
1227 : :
1228 : 330 : if (old_root)
1229 : : {
1230 : 21 : xaccAccountBeginEdit (old_root);
1231 : 21 : xaccAccountDestroy (old_root);
1232 : : }
1233 : : }
1234 : :
1235 : : Account *
1236 : 2487 : gnc_book_get_root_account (QofBook *book)
1237 : : {
1238 : : QofCollection *col;
1239 : : Account *root;
1240 : :
1241 : 2487 : if (!book) return nullptr;
1242 : 2486 : col = qof_book_get_collection (book, GNC_ID_ROOT_ACCOUNT);
1243 : 2486 : root = gnc_coll_get_root_account (col);
1244 : 2486 : if (root == nullptr && !qof_book_shutting_down(book))
1245 : 205 : root = gnc_account_create_root(book);
1246 : 2486 : return root;
1247 : : }
1248 : :
1249 : : void
1250 : 332 : gnc_book_set_root_account (QofBook *book, Account *root)
1251 : : {
1252 : : QofCollection *col;
1253 : 332 : if (!book) return;
1254 : :
1255 : 332 : if (root && gnc_account_get_book(root) != book)
1256 : : {
1257 : 1 : PERR ("cannot mix and match books freely!");
1258 : 1 : return;
1259 : : }
1260 : :
1261 : 331 : col = qof_book_get_collection (book, GNC_ID_ROOT_ACCOUNT);
1262 : 331 : gnc_coll_set_root_account (col, root);
1263 : : }
1264 : :
1265 : : /********************************************************************\
1266 : : \********************************************************************/
1267 : :
1268 : : Account *
1269 : 5044 : xaccMallocAccount (QofBook *book)
1270 : : {
1271 : : Account *acc;
1272 : :
1273 : 5044 : g_return_val_if_fail (book, nullptr);
1274 : :
1275 : 5044 : acc = static_cast<Account*>(g_object_new (GNC_TYPE_ACCOUNT, nullptr));
1276 : 5044 : xaccInitAccount (acc, book);
1277 : 5044 : qof_event_gen (&acc->inst, QOF_EVENT_CREATE, nullptr);
1278 : :
1279 : 5044 : return acc;
1280 : : }
1281 : :
1282 : : Account *
1283 : 288 : gnc_account_create_root (QofBook *book)
1284 : : {
1285 : : Account *root;
1286 : : AccountPrivate *rpriv;
1287 : :
1288 : 288 : root = xaccMallocAccount(book);
1289 : 288 : rpriv = GET_PRIVATE(root);
1290 : 288 : xaccAccountBeginEdit(root);
1291 : 288 : rpriv->type = ACCT_TYPE_ROOT;
1292 : 288 : rpriv->accountName = qof_string_cache_replace(rpriv->accountName, "Root Account");
1293 : 288 : mark_account (root);
1294 : 288 : xaccAccountCommitEdit(root);
1295 : 288 : gnc_book_set_root_account(book, root);
1296 : 288 : return root;
1297 : : }
1298 : :
1299 : : Account *
1300 : 3 : xaccCloneAccount(const Account *from, QofBook *book)
1301 : : {
1302 : : Account *ret;
1303 : : AccountPrivate *from_priv, *priv;
1304 : :
1305 : 3 : g_return_val_if_fail(GNC_IS_ACCOUNT(from), nullptr);
1306 : 2 : g_return_val_if_fail(QOF_IS_BOOK(book), nullptr);
1307 : :
1308 : 1 : ENTER (" ");
1309 : 1 : ret = static_cast<Account*>(g_object_new (GNC_TYPE_ACCOUNT, nullptr));
1310 : 1 : g_return_val_if_fail (ret, nullptr);
1311 : :
1312 : 1 : from_priv = GET_PRIVATE(from);
1313 : 1 : priv = GET_PRIVATE(ret);
1314 : 1 : xaccInitAccount (ret, book);
1315 : :
1316 : : /* Do not Begin/CommitEdit() here; give the caller
1317 : : * a chance to fix things up, and let them do it.
1318 : : * Also let caller issue the generate_event (EVENT_CREATE) */
1319 : 1 : priv->type = from_priv->type;
1320 : :
1321 : 1 : priv->accountName = qof_string_cache_replace(priv->accountName, from_priv->accountName);
1322 : 1 : priv->accountCode = qof_string_cache_replace(priv->accountCode, from_priv->accountCode);
1323 : 1 : priv->description = qof_string_cache_replace(priv->description, from_priv->description);
1324 : :
1325 : 1 : qof_instance_copy_kvp (QOF_INSTANCE (ret), QOF_INSTANCE (from));
1326 : :
1327 : : /* The new book should contain a commodity that matches
1328 : : * the one in the old book. Find it, use it. */
1329 : 1 : priv->commodity = gnc_commodity_obtain_twin(from_priv->commodity, book);
1330 : 1 : gnc_commodity_increment_usage_count(priv->commodity);
1331 : :
1332 : 1 : priv->commodity_scu = from_priv->commodity_scu;
1333 : 1 : priv->non_standard_scu = from_priv->non_standard_scu;
1334 : :
1335 : 1 : qof_instance_set_dirty(&ret->inst);
1336 : 1 : LEAVE (" ");
1337 : 1 : return ret;
1338 : : }
1339 : :
1340 : : /********************************************************************\
1341 : : \********************************************************************/
1342 : :
1343 : : static void
1344 : 358 : xaccFreeOneChildAccount (Account *acc)
1345 : : {
1346 : : /* FIXME: this code is kind of hacky. actually, all this code
1347 : : * seems to assume that the account edit levels are all 1. */
1348 : 358 : if (qof_instance_get_editlevel(acc) == 0)
1349 : 337 : xaccAccountBeginEdit(acc);
1350 : 358 : xaccAccountDestroy(acc);
1351 : 358 : }
1352 : :
1353 : : static void
1354 : 2208 : xaccFreeAccountChildren (Account *acc)
1355 : : {
1356 : 2208 : auto priv{GET_PRIVATE(acc)};
1357 : : /* Copy the list since it will be modified */
1358 : 2208 : auto children = priv->children;
1359 : 2208 : std::for_each (children.begin(), children.end(), xaccFreeOneChildAccount);
1360 : :
1361 : : /* The foreach should have removed all the children already. */
1362 : 2208 : priv->children.clear();
1363 : 2208 : }
1364 : :
1365 : : /* The xaccFreeAccount() routine releases memory associated with the
1366 : : * account. It should never be called directly from user code;
1367 : : * instead, the xaccAccountDestroy() routine should be used (because
1368 : : * xaccAccountDestroy() has the correct commit semantics). */
1369 : : static void
1370 : 2207 : xaccFreeAccount (Account *acc)
1371 : : {
1372 : : AccountPrivate *priv;
1373 : : GList *lp;
1374 : :
1375 : 2207 : g_return_if_fail(GNC_IS_ACCOUNT(acc));
1376 : :
1377 : 2207 : priv = GET_PRIVATE(acc);
1378 : 2207 : qof_event_gen (&acc->inst, QOF_EVENT_DESTROY, nullptr);
1379 : :
1380 : : /* Otherwise the lists below get munged while we're iterating
1381 : : * them, possibly crashing.
1382 : : */
1383 : 2207 : if (!qof_instance_get_destroying (acc))
1384 : 1 : qof_instance_set_destroying(acc, TRUE);
1385 : :
1386 : 2207 : if (!priv->children.empty())
1387 : : {
1388 : 1 : PERR (" instead of calling xaccFreeAccount(), please call\n"
1389 : : " xaccAccountBeginEdit(); xaccAccountDestroy();\n");
1390 : :
1391 : : /* First, recursively free children, also frees list */
1392 : 1 : xaccFreeAccountChildren(acc);
1393 : : }
1394 : :
1395 : : /* remove lots -- although these should be gone by now. */
1396 : 2207 : if (priv->lots)
1397 : : {
1398 : 1 : PERR (" instead of calling xaccFreeAccount(), please call\n"
1399 : : " xaccAccountBeginEdit(); xaccAccountDestroy();\n");
1400 : :
1401 : 4 : for (lp = priv->lots; lp; lp = lp->next)
1402 : : {
1403 : 3 : GNCLot *lot = static_cast<GNCLot*>(lp->data);
1404 : 3 : gnc_lot_destroy (lot);
1405 : : }
1406 : 1 : g_list_free (priv->lots);
1407 : 1 : priv->lots = nullptr;
1408 : : }
1409 : :
1410 : : /* Next, clean up the splits */
1411 : : /* NB there shouldn't be any splits by now ... they should
1412 : : * have been all been freed by CommitEdit(). We can remove this
1413 : : * check once we know the warning isn't occurring any more. */
1414 : 2207 : if (!priv->splits.empty())
1415 : : {
1416 : 2 : PERR (" instead of calling xaccFreeAccount(), please call\n"
1417 : : " xaccAccountBeginEdit(); xaccAccountDestroy();\n");
1418 : :
1419 : 2 : qof_instance_reset_editlevel(acc);
1420 : :
1421 : 8 : for (auto s : priv->splits)
1422 : : {
1423 : 6 : g_assert(xaccSplitGetAccount(s) == acc);
1424 : 6 : xaccSplitDestroy (s);
1425 : : }
1426 : : /* Nothing here (or in xaccAccountCommitEdit) nullptrs priv->splits, so this asserts every time.
1427 : : g_assert(priv->splits == nullptr);
1428 : : */
1429 : : }
1430 : :
1431 : 2207 : qof_string_cache_remove(priv->accountName);
1432 : 2207 : qof_string_cache_remove(priv->accountCode);
1433 : 2207 : qof_string_cache_remove(priv->description);
1434 : 2207 : priv->accountName = priv->accountCode = priv->description = nullptr;
1435 : :
1436 : : /* zero out values, just in case stray
1437 : : * pointers are pointing here. */
1438 : :
1439 : 2207 : priv->last_num = nullptr;
1440 : 2207 : priv->tax_us_code = nullptr;
1441 : 2207 : priv->tax_us_pns = nullptr;
1442 : 2207 : priv->color = nullptr;
1443 : 2207 : priv->sort_order = nullptr;
1444 : 2207 : priv->notes = nullptr;
1445 : 2207 : priv->filter = nullptr;
1446 : :
1447 : 2207 : priv->parent = nullptr;
1448 : :
1449 : 2207 : priv->balance = gnc_numeric_zero();
1450 : 2207 : priv->noclosing_balance = gnc_numeric_zero();
1451 : 2207 : priv->cleared_balance = gnc_numeric_zero();
1452 : 2207 : priv->reconciled_balance = gnc_numeric_zero();
1453 : :
1454 : 2207 : priv->type = ACCT_TYPE_NONE;
1455 : 2207 : gnc_commodity_decrement_usage_count(priv->commodity);
1456 : 2207 : priv->commodity = nullptr;
1457 : :
1458 : 2207 : priv->balance_dirty = FALSE;
1459 : 2207 : priv->sort_dirty = FALSE;
1460 : 2207 : priv->splits.~SplitsVec();
1461 : 2207 : priv->children.~AccountVec();
1462 : 2207 : g_hash_table_destroy (priv->splits_hash);
1463 : :
1464 : : /* qof_instance_release (&acc->inst); */
1465 : 2207 : g_object_unref(acc);
1466 : : }
1467 : :
1468 : : /********************************************************************\
1469 : : * transactional routines
1470 : : \********************************************************************/
1471 : :
1472 : : void
1473 : 34667 : xaccAccountBeginEdit (Account *acc)
1474 : : {
1475 : 34667 : g_return_if_fail(acc);
1476 : 34667 : qof_begin_edit(&acc->inst);
1477 : : }
1478 : :
1479 : 12536 : static void on_done(QofInstance *inst)
1480 : : {
1481 : : /* old event style */
1482 : 12536 : qof_event_gen (inst, QOF_EVENT_MODIFY, nullptr);
1483 : 12536 : }
1484 : :
1485 : 0 : static void on_err (QofInstance *inst, QofBackendError errcode)
1486 : : {
1487 : 0 : PERR("commit error: %d", errcode);
1488 : 0 : gnc_engine_signal_commit_error( errcode );
1489 : 0 : }
1490 : :
1491 : 2206 : static void acc_free (QofInstance *inst)
1492 : : {
1493 : : AccountPrivate *priv;
1494 : 2206 : Account *acc = (Account *) inst;
1495 : :
1496 : 2206 : priv = GET_PRIVATE(acc);
1497 : 2206 : if (priv->parent)
1498 : 1791 : gnc_account_remove_child(priv->parent, acc);
1499 : 2206 : xaccFreeAccount(acc);
1500 : 2206 : }
1501 : :
1502 : : static void
1503 : 78 : destroy_pending_splits_for_account(QofInstance *ent, gpointer acc)
1504 : : {
1505 : 78 : Transaction *trans = (Transaction *) ent;
1506 : : Split *split;
1507 : :
1508 : 78 : if (xaccTransIsOpen(trans))
1509 : 29 : while ((split = xaccTransFindSplitByAccount(trans, static_cast<Account*>(acc))))
1510 : 10 : xaccSplitDestroy(split);
1511 : 78 : }
1512 : :
1513 : : void
1514 : 33942 : xaccAccountCommitEdit (Account *acc)
1515 : : {
1516 : : AccountPrivate *priv;
1517 : : QofBook *book;
1518 : :
1519 : 33942 : g_return_if_fail(acc);
1520 : 33942 : if (!qof_commit_edit(&acc->inst)) return;
1521 : :
1522 : : /* If marked for deletion, get rid of subaccounts first,
1523 : : * and then the splits ... */
1524 : 14742 : priv = GET_PRIVATE(acc);
1525 : 14742 : if (qof_instance_get_destroying(acc))
1526 : : {
1527 : : QofCollection *col;
1528 : :
1529 : 2206 : qof_instance_increase_editlevel(acc);
1530 : :
1531 : : /* First, recursively free children */
1532 : 2206 : xaccFreeAccountChildren(acc);
1533 : :
1534 : 2206 : PINFO ("freeing splits for account %p (%s)",
1535 : : acc, priv->accountName ? priv->accountName : "(null)");
1536 : :
1537 : 2206 : book = qof_instance_get_book(acc);
1538 : :
1539 : : /* If book is shutting down, just clear the split list. The splits
1540 : : themselves will be destroyed by the transaction code */
1541 : 2206 : if (!qof_book_shutting_down(book))
1542 : : {
1543 : : // We need to delete in reverse order so that the vector's iterators aren't invalidated.
1544 : 516 : for_each(priv->splits.rbegin(), priv->splits.rend(), [](Split *s) {
1545 : 3 : xaccSplitDestroy (s); });
1546 : : }
1547 : : else
1548 : : {
1549 : 1690 : priv->splits.clear();
1550 : 1690 : g_hash_table_remove_all (priv->splits_hash);
1551 : : }
1552 : :
1553 : : /* It turns out there's a case where this assertion does not hold:
1554 : : When the user tries to delete an Imbalance account, while also
1555 : : deleting all the splits in it. The splits will just get
1556 : : recreated and put right back into the same account!
1557 : :
1558 : : g_assert(priv->splits == nullptr || qof_book_shutting_down(acc->inst.book));
1559 : : */
1560 : :
1561 : 2206 : if (!qof_book_shutting_down(book))
1562 : : {
1563 : 516 : col = qof_book_get_collection(book, GNC_ID_TRANS);
1564 : 516 : qof_collection_foreach(col, destroy_pending_splits_for_account, acc);
1565 : :
1566 : : /* the lots should be empty by now */
1567 : 519 : for (auto lp = priv->lots; lp; lp = lp->next)
1568 : : {
1569 : 3 : GNCLot *lot = static_cast<GNCLot*>(lp->data);
1570 : 3 : gnc_lot_destroy (lot);
1571 : : }
1572 : : }
1573 : 2206 : g_list_free(priv->lots);
1574 : 2206 : priv->lots = nullptr;
1575 : :
1576 : 2206 : qof_instance_set_dirty(&acc->inst);
1577 : 2206 : qof_instance_decrease_editlevel(acc);
1578 : : }
1579 : : else
1580 : : {
1581 : 12536 : xaccAccountBringUpToDate(acc);
1582 : : }
1583 : :
1584 : 14742 : qof_commit_edit_part2(&acc->inst, on_err, on_done, acc_free);
1585 : : }
1586 : :
1587 : : void
1588 : 2261 : xaccAccountDestroy (Account *acc)
1589 : : {
1590 : 2261 : g_return_if_fail(GNC_IS_ACCOUNT(acc));
1591 : :
1592 : 2261 : qof_instance_set_destroying(acc, TRUE);
1593 : :
1594 : 2261 : xaccAccountCommitEdit (acc);
1595 : : }
1596 : :
1597 : : void
1598 : 0 : xaccAccountDestroyAllTransactions(Account *acc)
1599 : : {
1600 : 0 : auto priv = GET_PRIVATE(acc);
1601 : 0 : std::vector<Transaction*> transactions;
1602 : 0 : transactions.reserve(priv->splits.size());
1603 : 0 : std::transform(priv->splits.begin(), priv->splits.end(),
1604 : : back_inserter(transactions),
1605 : 0 : [](auto split) { return split->parent; });
1606 : 0 : std::stable_sort(transactions.begin(), transactions.end());
1607 : 0 : transactions.erase(std::unique(transactions.begin(), transactions.end()),
1608 : 0 : transactions.end());
1609 : 0 : qof_event_suspend();
1610 : 0 : std::for_each(transactions.rbegin(), transactions.rend(),
1611 : 0 : [](auto trans) { xaccTransDestroy (trans); });
1612 : 0 : qof_event_resume();
1613 : 0 : }
1614 : :
1615 : : /********************************************************************\
1616 : : \********************************************************************/
1617 : :
1618 : : static gboolean
1619 : 42 : xaccAcctChildrenEqual(const AccountVec& na,
1620 : : const AccountVec& nb,
1621 : : gboolean check_guids)
1622 : : {
1623 : 42 : if (na.size() != nb.size())
1624 : : {
1625 : 0 : PINFO ("Accounts have different numbers of children");
1626 : 0 : return (FALSE);
1627 : : }
1628 : :
1629 : 57 : for (auto aa : na)
1630 : : {
1631 : 15 : auto it_b = std::find_if (nb.begin(), nb.end(), [aa](auto ab) -> bool
1632 : : {
1633 : 28 : if (!aa) return (!ab);
1634 : 28 : if (!ab) return false;
1635 : 28 : auto code_a{GET_PRIVATE(aa)->accountCode};
1636 : 28 : auto code_b{GET_PRIVATE(ab)->accountCode};
1637 : 28 : if ((code_a && *code_a) || (code_b && *code_b)) return !g_strcmp0 (code_a, code_b);
1638 : 24 : return !g_strcmp0 (GET_PRIVATE(aa)->accountName, GET_PRIVATE(ab)->accountName);
1639 : : });
1640 : :
1641 : 15 : if (it_b == nb.end())
1642 : : {
1643 : 0 : PINFO ("Unable to find matching child account.");
1644 : 0 : return FALSE;
1645 : : }
1646 : 15 : else if (auto ab = *it_b; !xaccAccountEqual(aa, ab, check_guids))
1647 : : {
1648 : : char sa[GUID_ENCODING_LENGTH + 1];
1649 : : char sb[GUID_ENCODING_LENGTH + 1];
1650 : :
1651 : 0 : guid_to_string_buff (xaccAccountGetGUID (aa), sa);
1652 : 0 : guid_to_string_buff (xaccAccountGetGUID (ab), sb);
1653 : :
1654 : 0 : PWARN ("accounts %s and %s differ", sa, sb);
1655 : :
1656 : 0 : return(FALSE);
1657 : : }
1658 : : }
1659 : :
1660 : 42 : return(TRUE);
1661 : : }
1662 : :
1663 : : gboolean
1664 : 43 : xaccAccountEqual(const Account *aa, const Account *ab, gboolean check_guids)
1665 : : {
1666 : : AccountPrivate *priv_aa, *priv_ab;
1667 : :
1668 : 43 : if (!aa && !ab) return TRUE;
1669 : :
1670 : 42 : g_return_val_if_fail(GNC_IS_ACCOUNT(aa), FALSE);
1671 : 42 : g_return_val_if_fail(GNC_IS_ACCOUNT(ab), FALSE);
1672 : :
1673 : 42 : priv_aa = GET_PRIVATE(aa);
1674 : 42 : priv_ab = GET_PRIVATE(ab);
1675 : 42 : if (priv_aa->type != priv_ab->type)
1676 : : {
1677 : 0 : PWARN ("types differ: %d vs %d", priv_aa->type, priv_ab->type);
1678 : 0 : return FALSE;
1679 : : }
1680 : :
1681 : 42 : if (g_strcmp0(priv_aa->accountName, priv_ab->accountName) != 0)
1682 : : {
1683 : 0 : PWARN ("names differ: %s vs %s", priv_aa->accountName, priv_ab->accountName);
1684 : 0 : return FALSE;
1685 : : }
1686 : :
1687 : 42 : if (g_strcmp0(priv_aa->accountCode, priv_ab->accountCode) != 0)
1688 : : {
1689 : 0 : PWARN ("codes differ: %s vs %s", priv_aa->accountCode, priv_ab->accountCode);
1690 : 0 : return FALSE;
1691 : : }
1692 : :
1693 : 42 : if (g_strcmp0(priv_aa->description, priv_ab->description) != 0)
1694 : : {
1695 : 0 : PWARN ("descriptions differ: %s vs %s", priv_aa->description, priv_ab->description);
1696 : 0 : return FALSE;
1697 : : }
1698 : :
1699 : 42 : if (!gnc_commodity_equal(priv_aa->commodity, priv_ab->commodity))
1700 : : {
1701 : 0 : PWARN ("commodities differ");
1702 : 0 : return FALSE;
1703 : : }
1704 : :
1705 : 42 : if (check_guids)
1706 : : {
1707 : 24 : if (qof_instance_guid_compare(aa, ab) != 0)
1708 : : {
1709 : 0 : PWARN ("GUIDs differ");
1710 : 0 : return FALSE;
1711 : : }
1712 : : }
1713 : :
1714 : 42 : if (qof_instance_compare_kvp (QOF_INSTANCE (aa), QOF_INSTANCE (ab)) != 0)
1715 : : {
1716 : : char *frame_a;
1717 : : char *frame_b;
1718 : :
1719 : 0 : frame_a = qof_instance_kvp_as_string (QOF_INSTANCE (aa));
1720 : 0 : frame_b = qof_instance_kvp_as_string (QOF_INSTANCE (ab));
1721 : :
1722 : 0 : PWARN ("kvp frames differ:\n%s\n\nvs\n\n%s", frame_a, frame_b);
1723 : :
1724 : 0 : g_free (frame_a);
1725 : 0 : g_free (frame_b);
1726 : :
1727 : 0 : return FALSE;
1728 : : }
1729 : :
1730 : 42 : if (!gnc_numeric_equal(priv_aa->starting_balance, priv_ab->starting_balance))
1731 : : {
1732 : : char *str_a;
1733 : : char *str_b;
1734 : :
1735 : 0 : str_a = gnc_numeric_to_string(priv_aa->starting_balance);
1736 : 0 : str_b = gnc_numeric_to_string(priv_ab->starting_balance);
1737 : :
1738 : 0 : PWARN ("starting balances differ: %s vs %s", str_a, str_b);
1739 : :
1740 : 0 : g_free (str_a);
1741 : 0 : g_free (str_b);
1742 : :
1743 : 0 : return FALSE;
1744 : : }
1745 : :
1746 : 42 : if (!gnc_numeric_equal(priv_aa->starting_noclosing_balance,
1747 : : priv_ab->starting_noclosing_balance))
1748 : : {
1749 : : char *str_a;
1750 : : char *str_b;
1751 : :
1752 : 0 : str_a = gnc_numeric_to_string(priv_aa->starting_noclosing_balance);
1753 : 0 : str_b = gnc_numeric_to_string(priv_ab->starting_noclosing_balance);
1754 : :
1755 : 0 : PWARN ("starting noclosing balances differ: %s vs %s", str_a, str_b);
1756 : :
1757 : 0 : g_free (str_a);
1758 : 0 : g_free (str_b);
1759 : :
1760 : 0 : return FALSE;
1761 : : }
1762 : 42 : if (!gnc_numeric_equal(priv_aa->starting_cleared_balance,
1763 : : priv_ab->starting_cleared_balance))
1764 : : {
1765 : : char *str_a;
1766 : : char *str_b;
1767 : :
1768 : 0 : str_a = gnc_numeric_to_string(priv_aa->starting_cleared_balance);
1769 : 0 : str_b = gnc_numeric_to_string(priv_ab->starting_cleared_balance);
1770 : :
1771 : 0 : PWARN ("starting cleared balances differ: %s vs %s", str_a, str_b);
1772 : :
1773 : 0 : g_free (str_a);
1774 : 0 : g_free (str_b);
1775 : :
1776 : 0 : return FALSE;
1777 : : }
1778 : :
1779 : 42 : if (!gnc_numeric_equal(priv_aa->starting_reconciled_balance,
1780 : : priv_ab->starting_reconciled_balance))
1781 : : {
1782 : : char *str_a;
1783 : : char *str_b;
1784 : :
1785 : 0 : str_a = gnc_numeric_to_string(priv_aa->starting_reconciled_balance);
1786 : 0 : str_b = gnc_numeric_to_string(priv_ab->starting_reconciled_balance);
1787 : :
1788 : 0 : PWARN ("starting reconciled balances differ: %s vs %s", str_a, str_b);
1789 : :
1790 : 0 : g_free (str_a);
1791 : 0 : g_free (str_b);
1792 : :
1793 : 0 : return FALSE;
1794 : : }
1795 : :
1796 : 42 : if (!gnc_numeric_equal(priv_aa->balance, priv_ab->balance))
1797 : : {
1798 : : char *str_a;
1799 : : char *str_b;
1800 : :
1801 : 0 : str_a = gnc_numeric_to_string(priv_aa->balance);
1802 : 0 : str_b = gnc_numeric_to_string(priv_ab->balance);
1803 : :
1804 : 0 : PWARN ("balances differ: %s vs %s", str_a, str_b);
1805 : :
1806 : 0 : g_free (str_a);
1807 : 0 : g_free (str_b);
1808 : :
1809 : 0 : return FALSE;
1810 : : }
1811 : :
1812 : 42 : if (!gnc_numeric_equal(priv_aa->noclosing_balance, priv_ab->noclosing_balance))
1813 : : {
1814 : : char *str_a;
1815 : : char *str_b;
1816 : :
1817 : 0 : str_a = gnc_numeric_to_string(priv_aa->noclosing_balance);
1818 : 0 : str_b = gnc_numeric_to_string(priv_ab->noclosing_balance);
1819 : :
1820 : 0 : PWARN ("noclosing balances differ: %s vs %s", str_a, str_b);
1821 : :
1822 : 0 : g_free (str_a);
1823 : 0 : g_free (str_b);
1824 : :
1825 : 0 : return FALSE;
1826 : : }
1827 : 42 : if (!gnc_numeric_equal(priv_aa->cleared_balance, priv_ab->cleared_balance))
1828 : : {
1829 : : char *str_a;
1830 : : char *str_b;
1831 : :
1832 : 0 : str_a = gnc_numeric_to_string(priv_aa->cleared_balance);
1833 : 0 : str_b = gnc_numeric_to_string(priv_ab->cleared_balance);
1834 : :
1835 : 0 : PWARN ("cleared balances differ: %s vs %s", str_a, str_b);
1836 : :
1837 : 0 : g_free (str_a);
1838 : 0 : g_free (str_b);
1839 : :
1840 : 0 : return FALSE;
1841 : : }
1842 : :
1843 : 42 : if (!gnc_numeric_equal(priv_aa->reconciled_balance, priv_ab->reconciled_balance))
1844 : : {
1845 : : char *str_a;
1846 : : char *str_b;
1847 : :
1848 : 0 : str_a = gnc_numeric_to_string(priv_aa->reconciled_balance);
1849 : 0 : str_b = gnc_numeric_to_string(priv_ab->reconciled_balance);
1850 : :
1851 : 0 : PWARN ("reconciled balances differ: %s vs %s", str_a, str_b);
1852 : :
1853 : 0 : g_free (str_a);
1854 : 0 : g_free (str_b);
1855 : :
1856 : 0 : return FALSE;
1857 : : }
1858 : :
1859 : : /* no parent; always compare downwards. */
1860 : :
1861 : 42 : if (!std::equal (priv_aa->splits.begin(), priv_aa->splits.end(),
1862 : : priv_ab->splits.begin(), priv_ab->splits.end(),
1863 : 15 : [check_guids](auto sa, auto sb)
1864 : 15 : { return xaccSplitEqual(sa, sb, check_guids, true, false); }))
1865 : : {
1866 : 0 : PWARN ("splits differ");
1867 : 0 : return false;
1868 : : }
1869 : :
1870 : 42 : if (!xaccAcctChildrenEqual(priv_aa->children, priv_ab->children, check_guids))
1871 : : {
1872 : 0 : PWARN ("children differ");
1873 : 0 : return FALSE;
1874 : : }
1875 : :
1876 : 42 : return(TRUE);
1877 : : }
1878 : :
1879 : : /********************************************************************\
1880 : : \********************************************************************/
1881 : : void
1882 : 35133 : gnc_account_set_sort_dirty (Account *acc)
1883 : : {
1884 : : AccountPrivate *priv;
1885 : :
1886 : 35133 : g_return_if_fail(GNC_IS_ACCOUNT(acc));
1887 : :
1888 : 35133 : if (qof_instance_get_destroying(acc))
1889 : 0 : return;
1890 : :
1891 : 35133 : priv = GET_PRIVATE(acc);
1892 : 35133 : priv->sort_dirty = TRUE;
1893 : : }
1894 : :
1895 : : void
1896 : 35133 : gnc_account_set_balance_dirty (Account *acc)
1897 : : {
1898 : : AccountPrivate *priv;
1899 : :
1900 : 35133 : g_return_if_fail(GNC_IS_ACCOUNT(acc));
1901 : :
1902 : 35133 : if (qof_instance_get_destroying(acc))
1903 : 0 : return;
1904 : :
1905 : 35133 : priv = GET_PRIVATE(acc);
1906 : 35133 : priv->balance_dirty = TRUE;
1907 : : }
1908 : :
1909 : 0 : void gnc_account_set_defer_bal_computation (Account *acc, gboolean defer)
1910 : : {
1911 : : AccountPrivate *priv;
1912 : :
1913 : 0 : g_return_if_fail (GNC_IS_ACCOUNT (acc));
1914 : :
1915 : 0 : if (qof_instance_get_destroying (acc))
1916 : 0 : return;
1917 : :
1918 : 0 : priv = GET_PRIVATE (acc);
1919 : 0 : priv->defer_bal_computation = defer;
1920 : : }
1921 : :
1922 : 0 : gboolean gnc_account_get_defer_bal_computation (Account *acc)
1923 : : {
1924 : : AccountPrivate *priv;
1925 : 0 : if (!acc)
1926 : 0 : return false;
1927 : 0 : priv = GET_PRIVATE (acc);
1928 : 0 : return priv->defer_bal_computation;
1929 : : }
1930 : :
1931 : :
1932 : : /********************************************************************\
1933 : : \********************************************************************/
1934 : :
1935 : 561555 : static bool split_cmp_less (const Split* a, const Split* b)
1936 : : {
1937 : 561555 : return xaccSplitOrder (a, b) < 0;
1938 : : }
1939 : :
1940 : : gboolean
1941 : 7381 : gnc_account_insert_split (Account *acc, Split *s)
1942 : : {
1943 : : AccountPrivate *priv;
1944 : :
1945 : 7381 : g_return_val_if_fail(GNC_IS_ACCOUNT(acc), FALSE);
1946 : 7380 : g_return_val_if_fail(GNC_IS_SPLIT(s), FALSE);
1947 : :
1948 : 7379 : priv = GET_PRIVATE(acc);
1949 : 7379 : if (!g_hash_table_add (priv->splits_hash, s))
1950 : 2 : return false;
1951 : :
1952 : 7377 : priv->splits.push_back (s);
1953 : :
1954 : 7377 : if (qof_instance_get_editlevel(acc) == 0)
1955 : 5440 : std::sort (priv->splits.begin(), priv->splits.end(), split_cmp_less);
1956 : : else
1957 : 1937 : priv->sort_dirty = true;
1958 : :
1959 : : //FIXME: find better event
1960 : 7377 : qof_event_gen (&acc->inst, QOF_EVENT_MODIFY, nullptr);
1961 : : /* Also send an event based on the account */
1962 : 7377 : qof_event_gen(&acc->inst, GNC_EVENT_ITEM_ADDED, s);
1963 : :
1964 : 7377 : priv->balance_dirty = TRUE;
1965 : : // DRH: Should the below be added? It is present in the delete path.
1966 : : // xaccAccountRecomputeBalance(acc);
1967 : 7377 : return TRUE;
1968 : : }
1969 : :
1970 : : gboolean
1971 : 9991 : gnc_account_remove_split (Account *acc, Split *s)
1972 : : {
1973 : : AccountPrivate *priv;
1974 : :
1975 : 9991 : g_return_val_if_fail(GNC_IS_ACCOUNT(acc), FALSE);
1976 : 9991 : g_return_val_if_fail(GNC_IS_SPLIT(s), FALSE);
1977 : :
1978 : 9991 : priv = GET_PRIVATE(acc);
1979 : :
1980 : 9991 : if (!g_hash_table_remove (priv->splits_hash, s))
1981 : 4936 : return false;
1982 : :
1983 : : // shortcut pruning the last element. this is the most common
1984 : : // remove_split operation during UI or book shutdown.
1985 : 5055 : if (s == priv->splits.back())
1986 : 5012 : priv->splits.pop_back();
1987 : : else
1988 : 43 : priv->splits.erase (std::remove (priv->splits.begin(), priv->splits.end(), s),
1989 : 86 : priv->splits.end());
1990 : :
1991 : : //FIXME: find better event type
1992 : 5055 : qof_event_gen(&acc->inst, QOF_EVENT_MODIFY, nullptr);
1993 : : // And send the account-based event, too
1994 : 5055 : qof_event_gen(&acc->inst, GNC_EVENT_ITEM_REMOVED, s);
1995 : :
1996 : 5055 : priv->balance_dirty = TRUE;
1997 : 5055 : xaccAccountRecomputeBalance(acc);
1998 : 5055 : return TRUE;
1999 : : }
2000 : :
2001 : : void
2002 : 15576 : xaccAccountSortSplits (Account *acc, gboolean force)
2003 : : {
2004 : : AccountPrivate *priv;
2005 : :
2006 : 15576 : g_return_if_fail(GNC_IS_ACCOUNT(acc));
2007 : :
2008 : 15576 : priv = GET_PRIVATE(acc);
2009 : 15576 : if (!priv->sort_dirty || (!force && qof_instance_get_editlevel(acc) > 0))
2010 : 11841 : return;
2011 : 3735 : std::sort (priv->splits.begin(), priv->splits.end(), split_cmp_less);
2012 : 3735 : priv->sort_dirty = FALSE;
2013 : 3735 : priv->balance_dirty = TRUE;
2014 : : }
2015 : :
2016 : : static void
2017 : 12536 : xaccAccountBringUpToDate(Account *acc)
2018 : : {
2019 : 12536 : if (!acc) return;
2020 : :
2021 : : /* if a re-sort happens here, then everything will update, so the
2022 : : cost basis and balance calls are no-ops */
2023 : 12536 : xaccAccountSortSplits(acc, FALSE);
2024 : 12536 : xaccAccountRecomputeBalance(acc);
2025 : : }
2026 : :
2027 : : /********************************************************************\
2028 : : \********************************************************************/
2029 : :
2030 : : void
2031 : 1308 : xaccAccountSetGUID (Account *acc, const GncGUID *guid)
2032 : : {
2033 : 1308 : g_return_if_fail(GNC_IS_ACCOUNT(acc));
2034 : 1308 : g_return_if_fail(guid);
2035 : :
2036 : : /* XXX this looks fishy and weird to me ... */
2037 : 1308 : PINFO("acct=%p", acc);
2038 : 1308 : xaccAccountBeginEdit (acc);
2039 : 1308 : qof_instance_set_guid (&acc->inst, guid);
2040 : 1308 : qof_instance_set_dirty(&acc->inst);
2041 : 1308 : xaccAccountCommitEdit (acc);
2042 : : }
2043 : :
2044 : : /********************************************************************\
2045 : : \********************************************************************/
2046 : :
2047 : : Account *
2048 : 4274 : xaccAccountLookup (const GncGUID *guid, QofBook *book)
2049 : : {
2050 : : QofCollection *col;
2051 : 4274 : if (!guid || !book) return nullptr;
2052 : 4274 : col = qof_book_get_collection (book, GNC_ID_ACCOUNT);
2053 : 4274 : return (Account *) qof_collection_lookup_entity (col, guid);
2054 : : }
2055 : :
2056 : : /********************************************************************\
2057 : : \********************************************************************/
2058 : :
2059 : : void
2060 : 0 : xaccAccountSetMark (Account *acc, short m)
2061 : : {
2062 : : AccountPrivate *priv;
2063 : :
2064 : 0 : g_return_if_fail(GNC_IS_ACCOUNT(acc));
2065 : :
2066 : 0 : priv = GET_PRIVATE(acc);
2067 : 0 : priv->mark = m;
2068 : : }
2069 : :
2070 : : void
2071 : 0 : xaccClearMark (Account *acc, short val)
2072 : : {
2073 : : Account *root;
2074 : :
2075 : 0 : g_return_if_fail(GNC_IS_ACCOUNT(acc));
2076 : :
2077 : 0 : root = gnc_account_get_root(acc);
2078 : 0 : xaccClearMarkDown(root ? root : acc, val);
2079 : : }
2080 : :
2081 : : void
2082 : 0 : xaccClearMarkDown (Account *acc, short val)
2083 : : {
2084 : : AccountPrivate *priv;
2085 : :
2086 : 0 : g_return_if_fail(GNC_IS_ACCOUNT(acc));
2087 : :
2088 : 0 : priv = GET_PRIVATE(acc);
2089 : 0 : priv->mark = val;
2090 : 0 : std::for_each (priv->children.begin(), priv->children.end(),
2091 : 0 : [val](auto acc){ xaccClearMarkDown(acc, val); });
2092 : : }
2093 : :
2094 : : /********************************************************************\
2095 : : \********************************************************************/
2096 : :
2097 : : GNCPolicy *
2098 : 355 : gnc_account_get_policy (Account *acc)
2099 : : {
2100 : 355 : g_return_val_if_fail(GNC_IS_ACCOUNT(acc), nullptr);
2101 : :
2102 : 355 : return GET_PRIVATE(acc)->policy;
2103 : : }
2104 : :
2105 : : void
2106 : 0 : gnc_account_set_policy (Account *acc, GNCPolicy *policy)
2107 : : {
2108 : : AccountPrivate *priv;
2109 : :
2110 : 0 : g_return_if_fail(GNC_IS_ACCOUNT(acc));
2111 : :
2112 : 0 : priv = GET_PRIVATE(acc);
2113 : 0 : priv->policy = policy ? policy : xaccGetFIFOPolicy();
2114 : : }
2115 : :
2116 : : /********************************************************************\
2117 : : \********************************************************************/
2118 : :
2119 : : void
2120 : 128 : xaccAccountRemoveLot (Account *acc, GNCLot *lot)
2121 : : {
2122 : : AccountPrivate *priv;
2123 : :
2124 : 128 : g_return_if_fail(GNC_IS_ACCOUNT(acc));
2125 : 128 : g_return_if_fail(GNC_IS_LOT(lot));
2126 : :
2127 : 128 : priv = GET_PRIVATE(acc);
2128 : 128 : g_return_if_fail(priv->lots);
2129 : :
2130 : 128 : ENTER ("(acc=%p, lot=%p)", acc, lot);
2131 : 128 : priv->lots = g_list_remove(priv->lots, lot);
2132 : 128 : qof_event_gen (QOF_INSTANCE(lot), QOF_EVENT_REMOVE, nullptr);
2133 : 128 : qof_event_gen (&acc->inst, QOF_EVENT_MODIFY, nullptr);
2134 : 128 : LEAVE ("(acc=%p, lot=%p)", acc, lot);
2135 : : }
2136 : :
2137 : : void
2138 : 232 : xaccAccountInsertLot (Account *acc, GNCLot *lot)
2139 : : {
2140 : : AccountPrivate *priv, *opriv;
2141 : 232 : Account * old_acc = nullptr;
2142 : : Account* lot_account;
2143 : :
2144 : : /* errors */
2145 : 232 : g_return_if_fail(GNC_IS_ACCOUNT(acc));
2146 : 232 : g_return_if_fail(GNC_IS_LOT(lot));
2147 : :
2148 : : /* optimizations */
2149 : 232 : lot_account = gnc_lot_get_account(lot);
2150 : 232 : if (lot_account == acc)
2151 : 1 : return;
2152 : :
2153 : 231 : ENTER ("(acc=%p, lot=%p)", acc, lot);
2154 : :
2155 : : /* pull it out of the old account */
2156 : 231 : if (lot_account)
2157 : : {
2158 : 1 : old_acc = lot_account;
2159 : 1 : opriv = GET_PRIVATE(old_acc);
2160 : 1 : opriv->lots = g_list_remove(opriv->lots, lot);
2161 : : }
2162 : :
2163 : 231 : priv = GET_PRIVATE(acc);
2164 : 231 : priv->lots = g_list_prepend(priv->lots, lot);
2165 : 231 : gnc_lot_set_account(lot, acc);
2166 : :
2167 : : /* Don't move the splits to the new account. The caller will do this
2168 : : * if appropriate, and doing it here will not work if we are being
2169 : : * called from gnc_book_close_period since xaccAccountInsertSplit
2170 : : * will try to balance capital gains and things aren't ready for that. */
2171 : :
2172 : 231 : qof_event_gen (QOF_INSTANCE(lot), QOF_EVENT_ADD, nullptr);
2173 : 231 : qof_event_gen (&acc->inst, QOF_EVENT_MODIFY, nullptr);
2174 : :
2175 : 231 : LEAVE ("(acc=%p, lot=%p)", acc, lot);
2176 : : }
2177 : :
2178 : : /********************************************************************\
2179 : : \********************************************************************/
2180 : : static void
2181 : 0 : xaccPreSplitMove (Split *split)
2182 : : {
2183 : 0 : xaccTransBeginEdit (xaccSplitGetParent (split));
2184 : 0 : }
2185 : :
2186 : : static void
2187 : 0 : xaccPostSplitMove (Split *split, Account *accto)
2188 : : {
2189 : : Transaction *trans;
2190 : :
2191 : 0 : xaccSplitSetAccount(split, accto);
2192 : 0 : xaccSplitSetAmount(split, split->amount);
2193 : 0 : trans = xaccSplitGetParent (split);
2194 : 0 : xaccTransCommitEdit (trans);
2195 : 0 : }
2196 : :
2197 : : void
2198 : 0 : xaccAccountMoveAllSplits (Account *accfrom, Account *accto)
2199 : : {
2200 : : AccountPrivate *from_priv;
2201 : :
2202 : : /* errors */
2203 : 0 : g_return_if_fail(GNC_IS_ACCOUNT(accfrom));
2204 : 0 : g_return_if_fail(GNC_IS_ACCOUNT(accto));
2205 : :
2206 : : /* optimizations */
2207 : 0 : from_priv = GET_PRIVATE(accfrom);
2208 : 0 : if (from_priv->splits.empty() || accfrom == accto)
2209 : 0 : return;
2210 : :
2211 : : /* check for book mix-up */
2212 : 0 : g_return_if_fail (qof_instance_books_equal(accfrom, accto));
2213 : 0 : ENTER ("(accfrom=%p, accto=%p)", accfrom, accto);
2214 : :
2215 : 0 : xaccAccountBeginEdit(accfrom);
2216 : 0 : xaccAccountBeginEdit(accto);
2217 : : /* Begin editing both accounts and all transactions in accfrom. */
2218 : 0 : std::for_each (from_priv->splits.begin(), from_priv->splits.end(), xaccPreSplitMove);
2219 : :
2220 : : /* Concatenate accfrom's lists of splits and lots to accto's lists. */
2221 : : //to_priv->splits = g_list_concat(to_priv->splits, from_priv->splits);
2222 : : //to_priv->lots = g_list_concat(to_priv->lots, from_priv->lots);
2223 : :
2224 : : /* Set appropriate flags. */
2225 : : //from_priv->balance_dirty = TRUE;
2226 : : //from_priv->sort_dirty = FALSE;
2227 : : //to_priv->balance_dirty = TRUE;
2228 : : //to_priv->sort_dirty = TRUE;
2229 : :
2230 : : /*
2231 : : * Change each split's account back pointer to accto.
2232 : : * Convert each split's amount to accto's commodity.
2233 : : * Commit to editing each transaction.
2234 : : */
2235 : 0 : auto splits = from_priv->splits;
2236 : 0 : std::for_each (splits.begin(), splits.end(), [accto](auto s){ xaccPostSplitMove (s, accto); });
2237 : :
2238 : : /* Finally empty accfrom. */
2239 : 0 : g_assert(from_priv->splits.empty());
2240 : 0 : g_assert(from_priv->lots == nullptr);
2241 : 0 : xaccAccountCommitEdit(accfrom);
2242 : 0 : xaccAccountCommitEdit(accto);
2243 : :
2244 : 0 : LEAVE ("(accfrom=%p, accto=%p)", accfrom, accto);
2245 : 0 : }
2246 : :
2247 : :
2248 : : /********************************************************************\
2249 : : * xaccAccountRecomputeBalance *
2250 : : * recomputes the partial balances and the current balance for *
2251 : : * this account. *
2252 : : * *
2253 : : * The way the computation is done depends on whether the partial *
2254 : : * balances are for a monetary account (bank, cash, etc.) or a *
2255 : : * certificate account (stock portfolio, mutual fund). For bank *
2256 : : * accounts, the invariant amount is the dollar amount. For share *
2257 : : * accounts, the invariant amount is the number of shares. For *
2258 : : * share accounts, the share price fluctuates, and the current *
2259 : : * value of such an account is the number of shares times the *
2260 : : * current share price. *
2261 : : * *
2262 : : * Part of the complexity of this computation stems from the fact *
2263 : : * xacc uses a double-entry system, meaning that one transaction *
2264 : : * appears in two accounts: one account is debited, and the other *
2265 : : * is credited. When the transaction represents a sale of shares, *
2266 : : * or a purchase of shares, some care must be taken to compute *
2267 : : * balances correctly. For a sale of shares, the stock account must*
2268 : : * be debited in shares, but the bank account must be credited *
2269 : : * in dollars. Thus, two different mechanisms must be used to *
2270 : : * compute balances, depending on account type. *
2271 : : * *
2272 : : * Args: account -- the account for which to recompute balances *
2273 : : * Return: void *
2274 : : \********************************************************************/
2275 : :
2276 : : void
2277 : 34862 : xaccAccountRecomputeBalance (Account * acc)
2278 : : {
2279 : : AccountPrivate *priv;
2280 : : gnc_numeric balance;
2281 : : gnc_numeric noclosing_balance;
2282 : : gnc_numeric cleared_balance;
2283 : : gnc_numeric reconciled_balance;
2284 : :
2285 : 56857 : if (nullptr == acc) return;
2286 : :
2287 : 34039 : priv = GET_PRIVATE(acc);
2288 : 34039 : if (qof_instance_get_editlevel(acc) > 0) return;
2289 : 31656 : if (!priv->balance_dirty || priv->defer_bal_computation) return;
2290 : 21426 : if (qof_instance_get_destroying(acc)) return;
2291 : 21426 : if (qof_book_shutting_down(qof_instance_get_book(acc))) return;
2292 : :
2293 : 12044 : balance = priv->starting_balance;
2294 : 12044 : noclosing_balance = priv->starting_noclosing_balance;
2295 : 12044 : cleared_balance = priv->starting_cleared_balance;
2296 : 12044 : reconciled_balance = priv->starting_reconciled_balance;
2297 : :
2298 : 12044 : PINFO ("acct=%s starting baln=%" G_GINT64_FORMAT "/%" G_GINT64_FORMAT,
2299 : : priv->accountName, balance.num, balance.denom);
2300 : 110892 : for (auto split : priv->splits)
2301 : : {
2302 : 98848 : gnc_numeric amt = xaccSplitGetAmount (split);
2303 : :
2304 : 98848 : balance = gnc_numeric_add_fixed(balance, amt);
2305 : :
2306 : 98848 : if (NREC != split->reconciled)
2307 : : {
2308 : 9957 : cleared_balance = gnc_numeric_add_fixed(cleared_balance, amt);
2309 : : }
2310 : :
2311 : 98848 : if (YREC == split->reconciled ||
2312 : 96916 : FREC == split->reconciled)
2313 : : {
2314 : : reconciled_balance =
2315 : 3723 : gnc_numeric_add_fixed(reconciled_balance, amt);
2316 : : }
2317 : :
2318 : 98848 : if (!(xaccTransGetIsClosingTxn (split->parent)))
2319 : 98523 : noclosing_balance = gnc_numeric_add_fixed(noclosing_balance, amt);
2320 : :
2321 : 98848 : split->balance = balance;
2322 : 98848 : split->noclosing_balance = noclosing_balance;
2323 : 98848 : split->cleared_balance = cleared_balance;
2324 : 98848 : split->reconciled_balance = reconciled_balance;
2325 : :
2326 : : }
2327 : :
2328 : 12044 : priv->balance = balance;
2329 : 12044 : priv->noclosing_balance = noclosing_balance;
2330 : 12044 : priv->cleared_balance = cleared_balance;
2331 : 12044 : priv->reconciled_balance = reconciled_balance;
2332 : 12044 : priv->balance_dirty = FALSE;
2333 : : }
2334 : :
2335 : : /********************************************************************\
2336 : : \********************************************************************/
2337 : :
2338 : : /* The sort order is used to implicitly define an
2339 : : * order for report generation */
2340 : :
2341 : : static int typeorder[NUM_ACCOUNT_TYPES] =
2342 : : {
2343 : : ACCT_TYPE_BANK, ACCT_TYPE_STOCK, ACCT_TYPE_MUTUAL, ACCT_TYPE_CURRENCY,
2344 : : ACCT_TYPE_CASH, ACCT_TYPE_ASSET, ACCT_TYPE_RECEIVABLE,
2345 : : ACCT_TYPE_CREDIT, ACCT_TYPE_LIABILITY, ACCT_TYPE_PAYABLE,
2346 : : ACCT_TYPE_INCOME, ACCT_TYPE_EXPENSE, ACCT_TYPE_EQUITY, ACCT_TYPE_TRADING
2347 : : };
2348 : :
2349 : : static int revorder[NUM_ACCOUNT_TYPES] =
2350 : : {
2351 : : -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1
2352 : : };
2353 : :
2354 : :
2355 : : int
2356 : 14490 : xaccAccountOrder (const Account *aa, const Account *ab)
2357 : : {
2358 : : AccountPrivate *priv_aa, *priv_ab;
2359 : : const char *da, *db;
2360 : : int ta, tb, result;
2361 : :
2362 : 14490 : if (aa == ab) return 0;
2363 : 14488 : if (!ab) return -1;
2364 : 14487 : if (!aa) return +1;
2365 : :
2366 : 14486 : priv_aa = GET_PRIVATE(aa);
2367 : 14486 : priv_ab = GET_PRIVATE(ab);
2368 : :
2369 : : /* sort on accountCode strings */
2370 : 14486 : da = priv_aa->accountCode;
2371 : 14486 : db = priv_ab->accountCode;
2372 : :
2373 : : /* Otherwise do a string sort */
2374 : 14486 : result = g_strcmp0 (da, db);
2375 : 14486 : if (result)
2376 : 41 : return result;
2377 : :
2378 : : /* if account-type-order array not initialized, initialize it */
2379 : : /* this will happen at most once during program invocation */
2380 : 14445 : if (-1 == revorder[0])
2381 : : {
2382 : : int i;
2383 : 400 : for (i = 0; i < NUM_ACCOUNT_TYPES; i++)
2384 : : {
2385 : 375 : revorder [typeorder[i]] = i;
2386 : : }
2387 : : }
2388 : :
2389 : : /* otherwise, sort on account type */
2390 : 14445 : ta = priv_aa->type;
2391 : 14445 : tb = priv_ab->type;
2392 : 14445 : ta = revorder[ta];
2393 : 14445 : tb = revorder[tb];
2394 : 14445 : if (ta < tb) return -1;
2395 : 13239 : if (ta > tb) return +1;
2396 : :
2397 : : /* otherwise, sort on accountName strings */
2398 : 8060 : da = priv_aa->accountName;
2399 : 8060 : db = priv_ab->accountName;
2400 : 8060 : result = safe_utf8_collate(da, db);
2401 : 8060 : if (result)
2402 : 7943 : return result;
2403 : :
2404 : : /* guarantee a stable sort */
2405 : 117 : return qof_instance_guid_compare(aa, ab);
2406 : : }
2407 : :
2408 : : static int
2409 : 0 : qof_xaccAccountOrder (const Account **aa, const Account **ab)
2410 : : {
2411 : 0 : return xaccAccountOrder(*aa, *ab);
2412 : : }
2413 : :
2414 : : /********************************************************************\
2415 : : \********************************************************************/
2416 : :
2417 : : void
2418 : 4359 : xaccAccountSetType (Account *acc, GNCAccountType tip)
2419 : : {
2420 : : AccountPrivate *priv;
2421 : :
2422 : : /* errors */
2423 : 4359 : g_return_if_fail(GNC_IS_ACCOUNT(acc));
2424 : 4359 : g_return_if_fail(tip < NUM_ACCOUNT_TYPES);
2425 : :
2426 : : /* optimizations */
2427 : 4359 : priv = GET_PRIVATE(acc);
2428 : 4359 : if (priv->type == tip)
2429 : 64 : return;
2430 : :
2431 : 4295 : xaccAccountBeginEdit(acc);
2432 : 4295 : priv->type = tip;
2433 : 4295 : priv->balance_dirty = TRUE; /* new type may affect balance computation */
2434 : 4295 : mark_account(acc);
2435 : 4295 : xaccAccountCommitEdit(acc);
2436 : : }
2437 : :
2438 : : void
2439 : 5383 : xaccAccountSetName (Account *acc, const char *str)
2440 : : {
2441 : : AccountPrivate *priv;
2442 : :
2443 : : /* errors */
2444 : 5383 : g_return_if_fail(GNC_IS_ACCOUNT(acc));
2445 : 5383 : g_return_if_fail(str);
2446 : :
2447 : : /* optimizations */
2448 : 5383 : priv = GET_PRIVATE(acc);
2449 : 5383 : if (g_strcmp0(str, priv->accountName) == 0)
2450 : 14 : return;
2451 : :
2452 : 5369 : xaccAccountBeginEdit(acc);
2453 : 5369 : priv->accountName = qof_string_cache_replace(priv->accountName, str);
2454 : 5369 : mark_account (acc);
2455 : 5369 : xaccAccountCommitEdit(acc);
2456 : : }
2457 : :
2458 : : void
2459 : 1244 : xaccAccountSetCode (Account *acc, const char *str)
2460 : : {
2461 : : AccountPrivate *priv;
2462 : :
2463 : : /* errors */
2464 : 1244 : g_return_if_fail(GNC_IS_ACCOUNT(acc));
2465 : :
2466 : : /* optimizations */
2467 : 1244 : priv = GET_PRIVATE(acc);
2468 : 1244 : if (g_strcmp0(str, priv->accountCode) == 0)
2469 : 89 : return;
2470 : :
2471 : 1155 : xaccAccountBeginEdit(acc);
2472 : 1155 : priv->accountCode = qof_string_cache_replace(priv->accountCode, str ? str : "");
2473 : 1155 : mark_account (acc);
2474 : 1155 : xaccAccountCommitEdit(acc);
2475 : : }
2476 : :
2477 : : void
2478 : 2165 : xaccAccountSetDescription (Account *acc, const char *str)
2479 : : {
2480 : : AccountPrivate *priv;
2481 : :
2482 : : /* errors */
2483 : 2165 : g_return_if_fail(GNC_IS_ACCOUNT(acc));
2484 : :
2485 : : /* optimizations */
2486 : 2165 : priv = GET_PRIVATE(acc);
2487 : 2165 : if (g_strcmp0(str, priv->description) == 0)
2488 : 638 : return;
2489 : :
2490 : 1527 : xaccAccountBeginEdit(acc);
2491 : 1527 : priv->description = qof_string_cache_replace(priv->description, str ? str : "");
2492 : 1527 : mark_account (acc);
2493 : 1527 : xaccAccountCommitEdit(acc);
2494 : : }
2495 : :
2496 : : static void
2497 : 6 : set_kvp_gnc_numeric_path (Account *acc, const std::vector<std::string>& path,
2498 : : std::optional<gnc_numeric> value)
2499 : : {
2500 : 6 : xaccAccountBeginEdit(acc);
2501 : 6 : qof_instance_set_path_kvp<gnc_numeric> (QOF_INSTANCE(acc), value, path);
2502 : 6 : xaccAccountCommitEdit(acc);
2503 : 6 : }
2504 : :
2505 : : static std::optional<gnc_numeric>
2506 : 11 : get_kvp_gnc_numeric_path (const Account *acc, const Path& path)
2507 : : {
2508 : 11 : return qof_instance_get_path_kvp<gnc_numeric> (QOF_INSTANCE(acc), path);
2509 : : }
2510 : :
2511 : : static void
2512 : 1825 : set_kvp_string_path (Account *acc, std::vector<std::string> const & path,
2513 : : const char *value)
2514 : : {
2515 : 1825 : std::optional<const char*> val;
2516 : 1825 : if (value && *value)
2517 : 37 : val = g_strdup(value);
2518 : :
2519 : 1825 : xaccAccountBeginEdit(acc);
2520 : 1825 : qof_instance_set_path_kvp<const char*> (QOF_INSTANCE(acc), val, path);
2521 : 1825 : xaccAccountCommitEdit(acc);
2522 : 1825 : }
2523 : :
2524 : : static const char*
2525 : 2858 : get_kvp_string_path (const Account *acc, const Path& path)
2526 : : {
2527 : 2858 : auto rv{qof_instance_get_path_kvp<const char*> (QOF_INSTANCE(acc), path)};
2528 : 5716 : return rv ? *rv : nullptr;
2529 : : }
2530 : :
2531 : : static void
2532 : 81 : set_kvp_account_path (Account* acc, const Path& path, const Account* kvp_account)
2533 : : {
2534 : 81 : std::optional<GncGUID*> val;
2535 : 81 : if (kvp_account)
2536 : 80 : val = guid_copy(xaccAccountGetGUID (kvp_account));
2537 : :
2538 : 81 : xaccAccountBeginEdit(acc);
2539 : 81 : qof_instance_set_path_kvp<GncGUID*> (QOF_INSTANCE(acc), val, path);
2540 : 81 : xaccAccountCommitEdit(acc);
2541 : 81 : }
2542 : :
2543 : : static Account*
2544 : 15 : get_kvp_account_path (const Account *acc, const Path& path)
2545 : : {
2546 : 15 : auto val{qof_instance_get_path_kvp<GncGUID*> (QOF_INSTANCE(acc), path)};
2547 : 30 : return val ? xaccAccountLookup (*val, gnc_account_get_book (acc)) : nullptr;
2548 : : }
2549 : :
2550 : : static void
2551 : 176 : set_kvp_boolean_path (Account *acc, const Path& path, gboolean option)
2552 : : {
2553 : 176 : set_kvp_string_path (acc, path, option ? "true" : nullptr);
2554 : 176 : }
2555 : :
2556 : : static gboolean
2557 : 261 : get_kvp_boolean_path (const Account *acc, const Path& path)
2558 : : {
2559 : 261 : auto slot{QOF_INSTANCE(acc)->kvp_data->get_slot(path)};
2560 : 261 : if (!slot) return false;
2561 : 10 : switch (slot->get_type())
2562 : : {
2563 : 1 : case KvpValueImpl::Type::INT64:
2564 : 1 : return slot->get<int64_t>() != 0;
2565 : 9 : case KvpValueImpl::Type::STRING:
2566 : 9 : return g_strcmp0 (slot->get<const char*>(), "true") == 0;
2567 : 0 : default:
2568 : 0 : return false;
2569 : : }
2570 : : }
2571 : :
2572 : : static void
2573 : 19 : set_kvp_int64_path (Account *acc, const Path& path, std::optional<gint64> value)
2574 : : {
2575 : 19 : xaccAccountBeginEdit(acc);
2576 : 19 : qof_instance_set_path_kvp<int64_t> (QOF_INSTANCE(acc), value, path);
2577 : 19 : xaccAccountCommitEdit(acc);
2578 : 19 : }
2579 : :
2580 : : static const std::optional<gint64>
2581 : 21 : get_kvp_int64_path (const Account *acc, const Path& path)
2582 : : {
2583 : 21 : return qof_instance_get_path_kvp<int64_t> (QOF_INSTANCE(acc), path);
2584 : : }
2585 : :
2586 : : static GncGUID*
2587 : 2 : get_kvp_guid_path (const Account *acc, const Path& path)
2588 : : {
2589 : 2 : auto val{qof_instance_get_path_kvp<GncGUID*> (QOF_INSTANCE(acc), path)};
2590 : 4 : return val ? guid_copy(*val) : nullptr;
2591 : : }
2592 : :
2593 : : void
2594 : 707 : xaccAccountSetColor (Account *acc, const char *str)
2595 : : {
2596 : 1414 : set_kvp_string_path (acc, {"color"}, str);
2597 : 707 : }
2598 : :
2599 : : void
2600 : 5 : xaccAccountSetFilter (Account *acc, const char *str)
2601 : : {
2602 : 10 : set_kvp_string_path (acc, {"filter"}, str);
2603 : 5 : }
2604 : :
2605 : : void
2606 : 5 : xaccAccountSetSortOrder (Account *acc, const char *str)
2607 : : {
2608 : 10 : set_kvp_string_path (acc, {"sort-order"}, str);
2609 : 5 : }
2610 : :
2611 : : void
2612 : 2 : xaccAccountSetSortReversed (Account *acc, gboolean sortreversed)
2613 : : {
2614 : 4 : set_kvp_boolean_path (acc, {"sort-reversed"}, sortreversed);
2615 : 2 : }
2616 : :
2617 : : static void
2618 : 1 : qofAccountSetParent (Account *acc, QofInstance *parent)
2619 : : {
2620 : : Account *parent_acc;
2621 : :
2622 : 1 : g_return_if_fail(GNC_IS_ACCOUNT(acc));
2623 : 1 : g_return_if_fail(GNC_IS_ACCOUNT(parent));
2624 : :
2625 : 1 : parent_acc = GNC_ACCOUNT(parent);
2626 : 1 : xaccAccountBeginEdit(acc);
2627 : 1 : xaccAccountBeginEdit(parent_acc);
2628 : 1 : gnc_account_append_child(parent_acc, acc);
2629 : 1 : mark_account (parent_acc);
2630 : 1 : mark_account (acc);
2631 : 1 : xaccAccountCommitEdit(acc);
2632 : 1 : xaccAccountCommitEdit(parent_acc);
2633 : : }
2634 : :
2635 : : void
2636 : 521 : xaccAccountSetNotes (Account *acc, const char *str)
2637 : : {
2638 : 1042 : set_kvp_string_path (acc, {"notes"}, str);
2639 : 521 : }
2640 : :
2641 : : void
2642 : 4 : xaccAccountSetOnlineID (Account *acc, const char *id)
2643 : : {
2644 : 4 : g_return_if_fail (GNC_IS_ACCOUNT(acc));
2645 : 12 : set_kvp_string_path (acc, {KEY_ONLINE_ID}, id);
2646 : 4 : }
2647 : :
2648 : :
2649 : : void
2650 : 71 : xaccAccountSetAssociatedAccount (Account *acc, const char *tag, const Account* assoc_acct)
2651 : : {
2652 : 71 : g_return_if_fail (GNC_IS_ACCOUNT(acc));
2653 : 70 : g_return_if_fail (tag && *tag);
2654 : :
2655 : 276 : set_kvp_account_path (acc, {"associated-account", tag}, assoc_acct);
2656 : 207 : }
2657 : :
2658 : : void
2659 : 3394 : xaccAccountSetCommodity (Account * acc, gnc_commodity * com)
2660 : : {
2661 : : AccountPrivate *priv;
2662 : :
2663 : : /* errors */
2664 : 3394 : g_return_if_fail(GNC_IS_ACCOUNT(acc));
2665 : 3394 : g_return_if_fail(GNC_IS_COMMODITY(com));
2666 : :
2667 : : /* optimizations */
2668 : 3199 : priv = GET_PRIVATE(acc);
2669 : 3199 : if (com == priv->commodity)
2670 : 47 : return;
2671 : :
2672 : 3152 : xaccAccountBeginEdit(acc);
2673 : 3152 : gnc_commodity_decrement_usage_count(priv->commodity);
2674 : 3152 : priv->commodity = com;
2675 : 3152 : gnc_commodity_increment_usage_count(com);
2676 : 3152 : priv->commodity_scu = gnc_commodity_get_fraction(com);
2677 : 3152 : priv->non_standard_scu = FALSE;
2678 : :
2679 : : /* iterate over splits */
2680 : 3161 : for (auto s : priv->splits)
2681 : : {
2682 : 9 : Transaction *trans = xaccSplitGetParent (s);
2683 : :
2684 : 9 : xaccTransBeginEdit (trans);
2685 : 9 : xaccSplitSetAmount (s, xaccSplitGetAmount(s));
2686 : 9 : xaccTransCommitEdit (trans);
2687 : : }
2688 : :
2689 : 3152 : priv->sort_dirty = TRUE; /* Not needed. */
2690 : 3152 : priv->balance_dirty = TRUE;
2691 : 3152 : mark_account (acc);
2692 : :
2693 : 3152 : xaccAccountCommitEdit(acc);
2694 : : }
2695 : :
2696 : : /*
2697 : : * Set the account scu and then check to see if it is the same as the
2698 : : * commodity scu. This function is called when parsing the data file
2699 : : * and is designed to catch cases where the two were accidentally set
2700 : : * to mismatched values in the past.
2701 : : */
2702 : : void
2703 : 905 : xaccAccountSetCommoditySCU (Account *acc, int scu)
2704 : : {
2705 : : AccountPrivate *priv;
2706 : :
2707 : 905 : g_return_if_fail(GNC_IS_ACCOUNT(acc));
2708 : :
2709 : 905 : priv = GET_PRIVATE(acc);
2710 : 905 : xaccAccountBeginEdit(acc);
2711 : 905 : priv->commodity_scu = scu;
2712 : 905 : if (scu != gnc_commodity_get_fraction(priv->commodity))
2713 : 100 : priv->non_standard_scu = TRUE;
2714 : 905 : mark_account(acc);
2715 : 905 : xaccAccountCommitEdit(acc);
2716 : : }
2717 : :
2718 : : int
2719 : 706 : xaccAccountGetCommoditySCUi (const Account * acc)
2720 : : {
2721 : 706 : g_return_val_if_fail(GNC_IS_ACCOUNT(acc), 0);
2722 : 706 : return GET_PRIVATE(acc)->commodity_scu;
2723 : : }
2724 : :
2725 : : int
2726 : 18216 : xaccAccountGetCommoditySCU (const Account * acc)
2727 : : {
2728 : : AccountPrivate *priv;
2729 : :
2730 : 18216 : g_return_val_if_fail(GNC_IS_ACCOUNT(acc), 0);
2731 : :
2732 : 18216 : priv = GET_PRIVATE(acc);
2733 : 18216 : if (priv->non_standard_scu || !priv->commodity)
2734 : 1805 : return priv->commodity_scu;
2735 : 16411 : return gnc_commodity_get_fraction(priv->commodity);
2736 : : }
2737 : :
2738 : : void
2739 : 46 : xaccAccountSetNonStdSCU (Account *acc, gboolean flag)
2740 : : {
2741 : : AccountPrivate *priv;
2742 : :
2743 : 46 : g_return_if_fail(GNC_IS_ACCOUNT(acc));
2744 : :
2745 : 46 : priv = GET_PRIVATE(acc);
2746 : 46 : if (priv->non_standard_scu == flag)
2747 : 42 : return;
2748 : 4 : xaccAccountBeginEdit(acc);
2749 : 4 : priv->non_standard_scu = flag;
2750 : 4 : mark_account (acc);
2751 : 4 : xaccAccountCommitEdit(acc);
2752 : : }
2753 : :
2754 : : gboolean
2755 : 251 : xaccAccountGetNonStdSCU (const Account * acc)
2756 : : {
2757 : 251 : g_return_val_if_fail(GNC_IS_ACCOUNT(acc), 0);
2758 : 251 : return GET_PRIVATE(acc)->non_standard_scu;
2759 : : }
2760 : :
2761 : : /********************************************************************\
2762 : : \********************************************************************/
2763 : : /* below follow the old, deprecated currency/security routines. */
2764 : :
2765 : : void
2766 : 0 : DxaccAccountSetCurrency (Account * acc, gnc_commodity * currency)
2767 : : {
2768 : 0 : if ((!acc) || (!currency)) return;
2769 : :
2770 : 0 : auto s = gnc_commodity_get_unique_name (currency);
2771 : 0 : set_kvp_string_path (acc, {"old-currency"}, s);
2772 : :
2773 : 0 : auto book = qof_instance_get_book(acc);
2774 : 0 : auto table = gnc_commodity_table_get_table (book);
2775 : 0 : auto commodity = gnc_commodity_table_lookup_unique (table, s);
2776 : :
2777 : 0 : if (!commodity)
2778 : 0 : gnc_commodity_table_insert (table, currency);
2779 : : }
2780 : :
2781 : : /********************************************************************\
2782 : : \********************************************************************/
2783 : :
2784 : : void
2785 : 18196 : gnc_account_foreach_descendant (const Account *acc, std::function<void(Account*)> account_cb)
2786 : : {
2787 : 18196 : g_return_if_fail (GNC_IS_ACCOUNT(acc));
2788 : :
2789 : : // children must be a vector copy instead of reference because
2790 : : // some callers e.g. xaccAccountTreeScrubLots will modify the
2791 : : // children
2792 : 18196 : auto children = GET_PRIVATE(acc)->children;
2793 : 34827 : for (auto child : children)
2794 : : {
2795 : 16631 : account_cb (child);
2796 : 16631 : gnc_account_foreach_descendant (child, account_cb);
2797 : : }
2798 : 18196 : }
2799 : :
2800 : : static void
2801 : 13137 : account_foreach_descendant_sorted (const Account *acc, std::function<void(Account*)> account_cb)
2802 : : {
2803 : 13137 : g_return_if_fail (GNC_IS_ACCOUNT(acc));
2804 : :
2805 : 13137 : auto children = GET_PRIVATE(acc)->children;
2806 : 13137 : std::sort (children.begin(), children.end(),
2807 : 13328 : [](auto a, auto b) { return xaccAccountOrder (a, b) < 0; });
2808 : :
2809 : 23897 : for (auto child : children)
2810 : : {
2811 : 10760 : account_cb (child);
2812 : 10760 : account_foreach_descendant_sorted (child, account_cb);
2813 : : }
2814 : 13137 : }
2815 : :
2816 : : void
2817 : 4340 : gnc_account_append_child (Account *new_parent, Account *child)
2818 : : {
2819 : : AccountPrivate *ppriv, *cpriv;
2820 : : Account *old_parent;
2821 : : QofCollection *col;
2822 : :
2823 : : /* errors */
2824 : 4340 : g_assert(GNC_IS_ACCOUNT(new_parent));
2825 : 4340 : g_assert(GNC_IS_ACCOUNT(child));
2826 : :
2827 : : /* optimizations */
2828 : 4340 : ppriv = GET_PRIVATE(new_parent);
2829 : 4340 : cpriv = GET_PRIVATE(child);
2830 : 4340 : old_parent = cpriv->parent;
2831 : 4340 : if (old_parent == new_parent)
2832 : 385 : return;
2833 : :
2834 : : // xaccAccountBeginEdit(new_parent);
2835 : 3955 : xaccAccountBeginEdit(child);
2836 : 3955 : if (old_parent)
2837 : : {
2838 : 16 : gnc_account_remove_child(old_parent, child);
2839 : :
2840 : 16 : if (!qof_instance_books_equal(old_parent, new_parent))
2841 : : {
2842 : : /* hack alert -- this implementation is not exactly correct.
2843 : : * If the entity tables are not identical, then the 'from' book
2844 : : * may have a different backend than the 'to' book. This means
2845 : : * that we should get the 'from' backend to destroy this account,
2846 : : * and the 'to' backend to save it. Right now, this is broken.
2847 : : *
2848 : : * A 'correct' implementation similar to this is in Period.c
2849 : : * except its for transactions ...
2850 : : *
2851 : : * Note also, we need to reparent the children to the new book as well.
2852 : : */
2853 : 1 : PWARN ("reparenting accounts across books is not correctly supported\n");
2854 : :
2855 : 1 : qof_event_gen (&child->inst, QOF_EVENT_DESTROY, nullptr);
2856 : 1 : col = qof_book_get_collection (qof_instance_get_book(new_parent),
2857 : : GNC_ID_ACCOUNT);
2858 : 1 : qof_collection_insert_entity (col, &child->inst);
2859 : 1 : qof_event_gen (&child->inst, QOF_EVENT_CREATE, nullptr);
2860 : : }
2861 : : }
2862 : 3955 : cpriv->parent = new_parent;
2863 : 3955 : ppriv->children.push_back (child);
2864 : 3955 : qof_instance_set_dirty(&new_parent->inst);
2865 : 3955 : qof_instance_set_dirty(&child->inst);
2866 : :
2867 : : /* Send events data. Warning: The call to commit_edit is also going
2868 : : * to send a MODIFY event. If the gtktreemodelfilter code gets the
2869 : : * MODIFY before it gets the ADD, it gets very confused and thinks
2870 : : * that two nodes have been added. */
2871 : 3955 : qof_event_gen (&child->inst, QOF_EVENT_ADD, nullptr);
2872 : : // qof_event_gen (&new_parent->inst, QOF_EVENT_MODIFY, nullptr);
2873 : :
2874 : 3955 : xaccAccountCommitEdit (child);
2875 : : // xaccAccountCommitEdit(new_parent);
2876 : : }
2877 : :
2878 : : void
2879 : 1839 : gnc_account_remove_child (Account *parent, Account *child)
2880 : : {
2881 : : AccountPrivate *ppriv, *cpriv;
2882 : : GncEventData ed;
2883 : :
2884 : 1840 : if (!child) return;
2885 : :
2886 : : /* Note this routine might be called on accounts which
2887 : : * are not yet parented. */
2888 : 1839 : if (!parent) return;
2889 : :
2890 : 1839 : ppriv = GET_PRIVATE(parent);
2891 : 1839 : cpriv = GET_PRIVATE(child);
2892 : :
2893 : 1839 : if (cpriv->parent != parent)
2894 : : {
2895 : 1 : PERR ("account not a child of parent");
2896 : 1 : return;
2897 : : }
2898 : :
2899 : : /* Gather event data */
2900 : 1838 : ed.node = parent;
2901 : 1838 : ed.idx = gnc_account_child_index (parent, child);
2902 : :
2903 : 1838 : ppriv->children.erase (std::remove (ppriv->children.begin(), ppriv->children.end(), child),
2904 : 1838 : ppriv->children.end());
2905 : :
2906 : : /* Now send the event. */
2907 : 1838 : qof_event_gen(&child->inst, QOF_EVENT_REMOVE, &ed);
2908 : :
2909 : : /* clear the account's parent pointer after REMOVE event generation. */
2910 : 1838 : cpriv->parent = nullptr;
2911 : :
2912 : 1838 : qof_event_gen (&parent->inst, QOF_EVENT_MODIFY, nullptr);
2913 : : }
2914 : :
2915 : : Account *
2916 : 36912 : gnc_account_get_parent (const Account *acc)
2917 : : {
2918 : 36912 : g_return_val_if_fail(GNC_IS_ACCOUNT(acc), nullptr);
2919 : 36912 : return GET_PRIVATE(acc)->parent;
2920 : : }
2921 : :
2922 : : Account *
2923 : 1603 : gnc_account_get_root (Account *acc)
2924 : : {
2925 : 1603 : g_return_val_if_fail(GNC_IS_ACCOUNT(acc), nullptr);
2926 : :
2927 : 4093 : while (auto parent = gnc_account_get_parent (acc))
2928 : 2490 : acc = parent;
2929 : :
2930 : 1603 : return acc;
2931 : : }
2932 : :
2933 : : gboolean
2934 : 38954 : gnc_account_is_root (const Account *account)
2935 : : {
2936 : 38954 : g_return_val_if_fail(GNC_IS_ACCOUNT(account), FALSE);
2937 : 38954 : return (GET_PRIVATE(account)->parent == nullptr);
2938 : : }
2939 : :
2940 : : GList *
2941 : 3021 : gnc_account_get_children (const Account *account)
2942 : : {
2943 : 3021 : g_return_val_if_fail(GNC_IS_ACCOUNT(account), nullptr);
2944 : 3021 : auto& children = GET_PRIVATE(account)->children;
2945 : 6042 : return std::accumulate (children.rbegin(), children.rend(), static_cast<GList*>(nullptr),
2946 : 3021 : g_list_prepend);
2947 : : }
2948 : :
2949 : : GList *
2950 : 1442 : gnc_account_get_children_sorted (const Account *account)
2951 : : {
2952 : 1442 : g_return_val_if_fail(GNC_IS_ACCOUNT(account), nullptr);
2953 : 1442 : return g_list_sort(gnc_account_get_children (account), (GCompareFunc)xaccAccountOrder);
2954 : : }
2955 : :
2956 : : gint
2957 : 0 : gnc_account_n_children (const Account *account)
2958 : : {
2959 : 0 : g_return_val_if_fail(GNC_IS_ACCOUNT(account), 0);
2960 : 0 : return GET_PRIVATE(account)->children.size();
2961 : : }
2962 : :
2963 : : gint
2964 : 1841 : gnc_account_child_index (const Account *parent, const Account *child)
2965 : : {
2966 : 1841 : g_return_val_if_fail(GNC_IS_ACCOUNT(parent), -1);
2967 : 1841 : g_return_val_if_fail(GNC_IS_ACCOUNT(child), -1);
2968 : 1841 : auto& children = GET_PRIVATE(parent)->children;
2969 : 1841 : auto find_it = std::find (children.begin(), children.end(), child);
2970 : 3680 : return find_it == children.end() ? -1 : std::distance (children.begin(), find_it);
2971 : : }
2972 : :
2973 : : Account *
2974 : 0 : gnc_account_nth_child (const Account *parent, gint num)
2975 : : {
2976 : 0 : g_return_val_if_fail(GNC_IS_ACCOUNT(parent), nullptr);
2977 : 0 : if ((size_t)num >= GET_PRIVATE(parent)->children.size())
2978 : 0 : return nullptr;
2979 : 0 : return static_cast<Account*>(GET_PRIVATE(parent)->children.at (num));
2980 : : }
2981 : :
2982 : : gint
2983 : 26 : gnc_account_n_descendants (const Account *account)
2984 : : {
2985 : 26 : int count {0};
2986 : 26 : gnc_account_foreach_descendant (account, [&count](auto acc){ count++; });
2987 : 26 : return count;
2988 : : }
2989 : :
2990 : : gint
2991 : 549 : gnc_account_get_current_depth (const Account *account)
2992 : : {
2993 : : AccountPrivate *priv;
2994 : 549 : int depth = 0;
2995 : :
2996 : 549 : g_return_val_if_fail(GNC_IS_ACCOUNT(account), 0);
2997 : :
2998 : 549 : priv = GET_PRIVATE(account);
2999 : 2263 : while (priv->parent && (priv->type != ACCT_TYPE_ROOT))
3000 : : {
3001 : 1714 : account = priv->parent;
3002 : 1714 : priv = GET_PRIVATE(account);
3003 : 1714 : depth++;
3004 : : }
3005 : :
3006 : 549 : return depth;
3007 : : }
3008 : :
3009 : : gint
3010 : 644 : gnc_account_get_tree_depth (const Account *account)
3011 : : {
3012 : : AccountPrivate *priv;
3013 : 644 : g_return_val_if_fail(GNC_IS_ACCOUNT(account), 0);
3014 : :
3015 : 644 : priv = GET_PRIVATE(account);
3016 : 644 : if (!priv->children.size())
3017 : 412 : return 1;
3018 : :
3019 : 232 : return 1 + std::accumulate (priv->children.begin(), priv->children.end(),
3020 : 556 : 0, [](auto a, auto b)
3021 : 788 : { return std::max (a, gnc_account_get_tree_depth (b)); });
3022 : : }
3023 : :
3024 : : GList *
3025 : 767 : gnc_account_get_descendants (const Account *account)
3026 : : {
3027 : 767 : GList* list = nullptr;
3028 : 12750 : gnc_account_foreach_descendant (account, [&list](auto a){ list = g_list_prepend (list, a); });
3029 : 1534 : return g_list_reverse (list);
3030 : : }
3031 : :
3032 : : GList *
3033 : 2377 : gnc_account_get_descendants_sorted (const Account *account)
3034 : : {
3035 : 2377 : GList* list = nullptr;
3036 : 13137 : account_foreach_descendant_sorted (account, [&list](auto a){ list = g_list_prepend (list, a); });
3037 : 4754 : return g_list_reverse (list);
3038 : : }
3039 : :
3040 : : // because gnc_account_lookup_by_name and gnc_account_lookup_by_code
3041 : : // are described in Account.h searching breadth-first until 4.6, and
3042 : : // accidentally modified to search depth-first from 4.7
3043 : : // onwards. Restore breath-first searching in 4.11 onwards to match
3044 : : // previous behaviour and function description in Account.h
3045 : : static gpointer
3046 : 3931 : account_foreach_descendant_breadthfirst_until (const Account *acc,
3047 : : AccountCb2 thunk,
3048 : : gpointer user_data)
3049 : : {
3050 : 3931 : g_return_val_if_fail (GNC_IS_ACCOUNT(acc), nullptr);
3051 : 3931 : g_return_val_if_fail (thunk, nullptr);
3052 : :
3053 : 3931 : auto& children{GET_PRIVATE(acc)->children};
3054 : :
3055 : 8060 : for (auto acc : children)
3056 : 4400 : if (auto result = thunk (acc, user_data))
3057 : 271 : return result;
3058 : :
3059 : 6878 : for (auto acc: children)
3060 : 3655 : if (auto result = account_foreach_descendant_breadthfirst_until (acc, thunk, user_data))
3061 : 437 : return result;
3062 : :
3063 : 3223 : return nullptr;
3064 : : }
3065 : :
3066 : : static gpointer
3067 : 4369 : is_acct_name (Account *account, gpointer user_data)
3068 : : {
3069 : 4369 : auto name {static_cast<gchar*>(user_data)};
3070 : 4369 : return (g_strcmp0 (name, xaccAccountGetName (account)) ? nullptr : account);
3071 : : }
3072 : :
3073 : : Account *
3074 : 268 : gnc_account_lookup_by_name (const Account *parent, const char * name)
3075 : : {
3076 : 268 : return (Account*)account_foreach_descendant_breadthfirst_until (parent, is_acct_name, (char*)name);
3077 : : }
3078 : :
3079 : : static gpointer
3080 : 31 : is_acct_code (Account *account, gpointer user_data)
3081 : : {
3082 : 31 : auto name {static_cast<gchar*>(user_data)};
3083 : 31 : return (g_strcmp0 (name, xaccAccountGetCode (account)) ? nullptr : account);
3084 : : }
3085 : :
3086 : : Account *
3087 : 8 : gnc_account_lookup_by_code (const Account *parent, const char * code)
3088 : : {
3089 : 8 : return (Account*)account_foreach_descendant_breadthfirst_until (parent, is_acct_code, (char*)code);
3090 : : }
3091 : :
3092 : : static gpointer
3093 : 0 : is_opening_balance_account (Account* account, gpointer data)
3094 : : {
3095 : 0 : gnc_commodity* commodity = GNC_COMMODITY(data);
3096 : 0 : if (xaccAccountGetIsOpeningBalance(account) && gnc_commodity_equiv(commodity, xaccAccountGetCommodity(account)))
3097 : 0 : return account;
3098 : 0 : return nullptr;
3099 : : }
3100 : :
3101 : : Account*
3102 : 0 : gnc_account_lookup_by_opening_balance (Account* account, gnc_commodity* commodity)
3103 : : {
3104 : 0 : return (Account *)gnc_account_foreach_descendant_until (account, is_opening_balance_account, commodity);
3105 : : }
3106 : :
3107 : : /********************************************************************\
3108 : : * Fetch an account, given its full name *
3109 : : \********************************************************************/
3110 : :
3111 : : static Account *
3112 : 43 : gnc_account_lookup_by_full_name_helper (const Account *parent,
3113 : : gchar **names)
3114 : : {
3115 : 43 : g_return_val_if_fail(GNC_IS_ACCOUNT(parent), nullptr);
3116 : 43 : g_return_val_if_fail(names, nullptr);
3117 : :
3118 : : /* Look for the first name in the children. */
3119 : 117 : for (auto account : GET_PRIVATE(parent)->children)
3120 : : {
3121 : 104 : auto priv = GET_PRIVATE(account);
3122 : 104 : if (g_strcmp0(priv->accountName, names[0]) == 0)
3123 : : {
3124 : : /* We found an account. If the next entry is nullptr, there is
3125 : : * nothing left in the name, so just return the account. */
3126 : 35 : if (names[1] == nullptr)
3127 : 30 : return account;
3128 : :
3129 : : /* No children? We're done. */
3130 : 22 : if (priv->children.empty())
3131 : 0 : return nullptr;
3132 : :
3133 : : /* There's stuff left to search for. Search recursively. */
3134 : 22 : if (auto found = gnc_account_lookup_by_full_name_helper(account, &names[1]))
3135 : 17 : return found;
3136 : : }
3137 : : }
3138 : :
3139 : 13 : return nullptr;
3140 : : }
3141 : :
3142 : :
3143 : : Account *
3144 : 18 : gnc_account_lookup_by_full_name (const Account *any_acc,
3145 : : const gchar *name)
3146 : : {
3147 : : const AccountPrivate *rpriv;
3148 : : const Account *root;
3149 : : Account *found;
3150 : : gchar **names;
3151 : :
3152 : 18 : g_return_val_if_fail(GNC_IS_ACCOUNT(any_acc), nullptr);
3153 : 18 : g_return_val_if_fail(name, nullptr);
3154 : :
3155 : 18 : root = any_acc;
3156 : 18 : rpriv = GET_PRIVATE(root);
3157 : 30 : while (rpriv->parent)
3158 : : {
3159 : 12 : root = rpriv->parent;
3160 : 12 : rpriv = GET_PRIVATE(root);
3161 : : }
3162 : 18 : names = g_strsplit(name, gnc_get_account_separator_string(), -1);
3163 : 18 : found = gnc_account_lookup_by_full_name_helper(root, names);
3164 : 18 : g_strfreev(names);
3165 : 18 : return found;
3166 : : }
3167 : :
3168 : : GList*
3169 : 168 : gnc_account_lookup_by_type_and_commodity (Account* root,
3170 : : const char* name,
3171 : : GNCAccountType acctype,
3172 : : gnc_commodity* commodity)
3173 : : {
3174 : 168 : GList *retval{};
3175 : 168 : auto rpriv{GET_PRIVATE(root)};
3176 : 278 : for (auto account : rpriv->children)
3177 : : {
3178 : 110 : if (xaccAccountGetType (account) == acctype)
3179 : : {
3180 : 114 : if (commodity &&
3181 : 57 : !gnc_commodity_equiv(xaccAccountGetCommodity (account),
3182 : : commodity))
3183 : 10 : continue;
3184 : :
3185 : 47 : if (name && strcmp(name, xaccAccountGetName(account)))
3186 : 24 : continue;
3187 : :
3188 : 23 : retval = g_list_prepend(retval, account);
3189 : : }
3190 : : }
3191 : :
3192 : 168 : if (!retval) // Recurse through the children
3193 : 208 : for (auto account : rpriv->children)
3194 : : {
3195 : 63 : auto result = gnc_account_lookup_by_type_and_commodity(account,
3196 : : name,
3197 : : acctype,
3198 : : commodity);
3199 : 63 : if (result)
3200 : 0 : retval = g_list_concat(result, retval);
3201 : : }
3202 : 168 : return retval;
3203 : : }
3204 : :
3205 : : void
3206 : 150 : gnc_account_foreach_child (const Account *acc,
3207 : : AccountCb thunk,
3208 : : gpointer user_data)
3209 : : {
3210 : 150 : g_return_if_fail(GNC_IS_ACCOUNT(acc));
3211 : 150 : g_return_if_fail(thunk);
3212 : 150 : std::for_each (GET_PRIVATE(acc)->children.begin(), GET_PRIVATE(acc)->children.end(),
3213 : 53 : [user_data, thunk](auto a){ thunk (a, user_data); });
3214 : : }
3215 : :
3216 : : void
3217 : 715 : gnc_account_foreach_descendant (const Account *acc,
3218 : : AccountCb thunk,
3219 : : gpointer user_data)
3220 : : {
3221 : 3571 : gnc_account_foreach_descendant (acc, [&](auto acc){ thunk (acc, user_data); });
3222 : 715 : }
3223 : :
3224 : : gpointer
3225 : 91 : gnc_account_foreach_descendant_until (const Account *acc,
3226 : : AccountCb2 thunk,
3227 : : gpointer user_data)
3228 : : {
3229 : 91 : gpointer result {nullptr};
3230 : :
3231 : 91 : g_return_val_if_fail (GNC_IS_ACCOUNT(acc), nullptr);
3232 : 91 : g_return_val_if_fail (thunk, nullptr);
3233 : :
3234 : 156 : for (auto child : GET_PRIVATE(acc)->children)
3235 : : {
3236 : 87 : result = thunk (child, user_data);
3237 : 87 : if (result) break;
3238 : :
3239 : 80 : result = gnc_account_foreach_descendant_until (child, thunk, user_data);
3240 : 80 : if (result) break;
3241 : : }
3242 : :
3243 : 91 : return result;
3244 : : }
3245 : :
3246 : :
3247 : : GNCAccountType
3248 : 16294 : xaccAccountGetType (const Account *acc)
3249 : : {
3250 : 16294 : g_return_val_if_fail(GNC_IS_ACCOUNT(acc), ACCT_TYPE_NONE);
3251 : 16294 : return GET_PRIVATE(acc)->type;
3252 : : }
3253 : :
3254 : : static const char*
3255 : 40 : qofAccountGetTypeString (const Account *acc)
3256 : : {
3257 : 40 : g_return_val_if_fail(GNC_IS_ACCOUNT(acc), nullptr);
3258 : 40 : return xaccAccountTypeEnumAsString(GET_PRIVATE(acc)->type);
3259 : : }
3260 : :
3261 : : static void
3262 : 42 : qofAccountSetType (Account *acc, const char *type_string)
3263 : : {
3264 : 42 : g_return_if_fail(GNC_IS_ACCOUNT(acc));
3265 : 42 : g_return_if_fail(type_string);
3266 : 42 : xaccAccountSetType(acc, xaccAccountStringToEnum(type_string));
3267 : : }
3268 : :
3269 : : const char *
3270 : 72576 : xaccAccountGetName (const Account *acc)
3271 : : {
3272 : 72576 : g_return_val_if_fail(GNC_IS_ACCOUNT(acc), nullptr);
3273 : 72576 : return GET_PRIVATE(acc)->accountName;
3274 : : }
3275 : :
3276 : : std::vector<const Account*>
3277 : 10383 : gnc_account_get_all_parents (const Account *account)
3278 : : {
3279 : 10383 : std::vector<const Account*> rv;
3280 : 38707 : for (auto a = account; !gnc_account_is_root (a); a = gnc_account_get_parent (a))
3281 : 28324 : rv.push_back (a);
3282 : 10383 : return rv;
3283 : 0 : }
3284 : :
3285 : : gchar *
3286 : 10180 : gnc_account_get_full_name(const Account *account)
3287 : : {
3288 : : /* So much for hardening the API. Too many callers to this function don't
3289 : : * bother to check if they have a non-nullptr pointer before calling. */
3290 : 10180 : if (nullptr == account)
3291 : 1 : return g_strdup("");
3292 : :
3293 : : /* errors */
3294 : 10179 : g_return_val_if_fail(GNC_IS_ACCOUNT(account), g_strdup(""));
3295 : :
3296 : 10179 : auto path{gnc_account_get_all_parents (account)};
3297 : 10179 : auto seps_size{path.empty() ? 0 : strlen (account_separator) * (path.size() - 1)};
3298 : 10179 : auto alloc_size{std::accumulate (path.begin(), path.end(), seps_size,
3299 : 27859 : [](auto sum, auto acc)
3300 : 27859 : { return sum + strlen (xaccAccountGetName (acc)); })};
3301 : 10179 : auto rv = g_new (char, alloc_size + 1);
3302 : 10179 : auto p = rv;
3303 : :
3304 : 10179 : std::for_each (path.rbegin(), path.rend(),
3305 : 27859 : [&p, rv](auto a)
3306 : : {
3307 : 27859 : if (p != rv)
3308 : 17682 : p = stpcpy (p, account_separator);
3309 : 27859 : p = stpcpy (p, xaccAccountGetName (a));
3310 : 27859 : });
3311 : 10179 : *p = '\0';
3312 : :
3313 : 10179 : return rv;
3314 : 10179 : }
3315 : :
3316 : : const char *
3317 : 1982 : xaccAccountGetCode (const Account *acc)
3318 : : {
3319 : 1982 : g_return_val_if_fail(GNC_IS_ACCOUNT(acc), nullptr);
3320 : 1982 : return GET_PRIVATE(acc)->accountCode;
3321 : : }
3322 : :
3323 : : const char *
3324 : 1699 : xaccAccountGetDescription (const Account *acc)
3325 : : {
3326 : 1699 : g_return_val_if_fail(GNC_IS_ACCOUNT(acc), nullptr);
3327 : 1699 : return GET_PRIVATE(acc)->description;
3328 : : }
3329 : :
3330 : : const char *
3331 : 8 : xaccAccountGetColor (const Account *acc)
3332 : : {
3333 : 24 : return get_kvp_string_path (acc, {"color"});
3334 : : }
3335 : :
3336 : : const char *
3337 : 6 : xaccAccountGetFilter (const Account *acc)
3338 : : {
3339 : 18 : return get_kvp_string_path (acc, {"filter"});
3340 : : }
3341 : :
3342 : : const char *
3343 : 6 : xaccAccountGetSortOrder (const Account *acc)
3344 : : {
3345 : 18 : return get_kvp_string_path (acc, {"sort-order"});
3346 : : }
3347 : :
3348 : : gboolean
3349 : 3 : xaccAccountGetSortReversed (const Account *acc)
3350 : : {
3351 : 9 : return get_kvp_boolean_path (acc, {"sort-reversed"});
3352 : : }
3353 : :
3354 : : const char *
3355 : 1339 : xaccAccountGetNotes (const Account *acc)
3356 : : {
3357 : 4017 : return get_kvp_string_path (acc, {"notes"});
3358 : : }
3359 : :
3360 : : const char *
3361 : 81 : xaccAccountGetOnlineID (const Account *acc)
3362 : : {
3363 : 81 : g_return_val_if_fail (GNC_IS_ACCOUNT(acc), nullptr);
3364 : 243 : return get_kvp_string_path (acc, {KEY_ONLINE_ID});
3365 : 81 : }
3366 : :
3367 : : Account*
3368 : 3 : xaccAccountGetAssociatedAccount (const Account *acc, const char *tag)
3369 : : {
3370 : 3 : g_return_val_if_fail (tag && *tag, nullptr);
3371 : :
3372 : 12 : return get_kvp_account_path (acc, {"associated-account", tag});
3373 : 9 : }
3374 : :
3375 : :
3376 : : gnc_commodity *
3377 : 1363 : DxaccAccountGetCurrency (const Account *acc)
3378 : : {
3379 : 4089 : if (auto s = get_kvp_string_path (acc, {"old-currency"}))
3380 : : {
3381 : 0 : auto table = gnc_commodity_table_get_table (qof_instance_get_book(acc));
3382 : 0 : return gnc_commodity_table_lookup_unique (table, s);
3383 : : }
3384 : :
3385 : 1363 : return nullptr;
3386 : : }
3387 : :
3388 : : gnc_commodity *
3389 : 84182 : xaccAccountGetCommodity (const Account *acc)
3390 : : {
3391 : 84182 : if (!GNC_IS_ACCOUNT(acc))
3392 : 0 : return nullptr;
3393 : 84182 : return GET_PRIVATE(acc)->commodity;
3394 : : }
3395 : :
3396 : 427 : gnc_commodity * gnc_account_get_currency_or_parent(const Account* account)
3397 : : {
3398 : 427 : g_return_val_if_fail (GNC_IS_ACCOUNT (account), nullptr);
3399 : :
3400 : 856 : for (auto acc = account; acc; acc = gnc_account_get_parent (acc))
3401 : 853 : if (auto comm = xaccAccountGetCommodity (acc); gnc_commodity_is_currency (comm))
3402 : 424 : return comm;
3403 : :
3404 : 3 : return nullptr; // no suitable commodity found.
3405 : : }
3406 : :
3407 : : /********************************************************************\
3408 : : \********************************************************************/
3409 : : void
3410 : 1 : gnc_account_set_start_balance (Account *acc, const gnc_numeric start_baln)
3411 : : {
3412 : : AccountPrivate *priv;
3413 : :
3414 : 1 : g_return_if_fail(GNC_IS_ACCOUNT(acc));
3415 : :
3416 : 1 : priv = GET_PRIVATE(acc);
3417 : 1 : priv->starting_balance = start_baln;
3418 : 1 : priv->balance_dirty = TRUE;
3419 : : }
3420 : :
3421 : : void
3422 : 0 : gnc_account_set_start_cleared_balance (Account *acc,
3423 : : const gnc_numeric start_baln)
3424 : : {
3425 : : AccountPrivate *priv;
3426 : :
3427 : 0 : g_return_if_fail(GNC_IS_ACCOUNT(acc));
3428 : :
3429 : 0 : priv = GET_PRIVATE(acc);
3430 : 0 : priv->starting_cleared_balance = start_baln;
3431 : 0 : priv->balance_dirty = TRUE;
3432 : : }
3433 : :
3434 : : void
3435 : 0 : gnc_account_set_start_reconciled_balance (Account *acc,
3436 : : const gnc_numeric start_baln)
3437 : : {
3438 : : AccountPrivate *priv;
3439 : :
3440 : 0 : g_return_if_fail(GNC_IS_ACCOUNT(acc));
3441 : :
3442 : 0 : priv = GET_PRIVATE(acc);
3443 : 0 : priv->starting_reconciled_balance = start_baln;
3444 : 0 : priv->balance_dirty = TRUE;
3445 : : }
3446 : :
3447 : : gnc_numeric
3448 : 68 : xaccAccountGetBalance (const Account *acc)
3449 : : {
3450 : 68 : g_return_val_if_fail(GNC_IS_ACCOUNT(acc), gnc_numeric_zero());
3451 : 68 : return GET_PRIVATE(acc)->balance;
3452 : : }
3453 : :
3454 : : gnc_numeric
3455 : 38 : xaccAccountGetClearedBalance (const Account *acc)
3456 : : {
3457 : 38 : g_return_val_if_fail(GNC_IS_ACCOUNT(acc), gnc_numeric_zero());
3458 : 38 : return GET_PRIVATE(acc)->cleared_balance;
3459 : : }
3460 : :
3461 : : gnc_numeric
3462 : 20 : xaccAccountGetReconciledBalance (const Account *acc)
3463 : : {
3464 : 20 : g_return_val_if_fail(GNC_IS_ACCOUNT(acc), gnc_numeric_zero());
3465 : 20 : return GET_PRIVATE(acc)->reconciled_balance;
3466 : : }
3467 : :
3468 : : gnc_numeric
3469 : 2 : xaccAccountGetProjectedMinimumBalance (const Account *acc)
3470 : : {
3471 : 2 : auto today{gnc_time64_get_today_end()};
3472 : 2 : std::optional<gnc_numeric> minimum;
3473 : :
3474 : 6 : auto before_today_end = [&minimum, today](const Split *s) -> bool
3475 : : {
3476 : 6 : auto bal{xaccSplitGetBalance(s)};
3477 : 6 : if (!minimum || gnc_numeric_compare (bal, *minimum) < 0)
3478 : 3 : minimum = bal;
3479 : 12 : return (xaccTransGetDate(xaccSplitGetParent(s)) < today);
3480 : 2 : };
3481 : : // scan to find today's split, but we're really interested in the
3482 : : // minimum balance
3483 : 2 : [[maybe_unused]] auto todays_split = gnc_account_find_split (acc, before_today_end, true);
3484 : 4 : return minimum ? *minimum : gnc_numeric_zero();
3485 : : }
3486 : :
3487 : :
3488 : : /********************************************************************\
3489 : : \********************************************************************/
3490 : :
3491 : : static gnc_numeric
3492 : 2934 : GetBalanceAsOfDate (Account *acc, time64 date, std::function<gnc_numeric(Split*)> split_to_numeric)
3493 : : {
3494 : 2934 : g_return_val_if_fail(GNC_IS_ACCOUNT(acc), gnc_numeric_zero());
3495 : :
3496 : 2934 : xaccAccountSortSplits (acc, TRUE); /* just in case, normally a noop */
3497 : 2934 : xaccAccountRecomputeBalance (acc); /* just in case, normally a noop */
3498 : :
3499 : 4202 : auto is_before_date = [date](auto s) -> bool
3500 : 4202 : { return xaccTransGetDate(xaccSplitGetParent(s)) < date; };
3501 : :
3502 : 2934 : auto latest_split{gnc_account_find_split (acc, is_before_date, true)};
3503 : 2934 : return latest_split ? split_to_numeric (latest_split) : gnc_numeric_zero();
3504 : : }
3505 : :
3506 : : gnc_numeric
3507 : 372 : xaccAccountGetBalanceAsOfDate (Account *acc, time64 date)
3508 : : {
3509 : 372 : return GetBalanceAsOfDate (acc, date, xaccSplitGetBalance);
3510 : : }
3511 : :
3512 : : static gnc_numeric
3513 : 0 : xaccAccountGetNoclosingBalanceAsOfDate (Account *acc, time64 date)
3514 : : {
3515 : 0 : return GetBalanceAsOfDate (acc, date, xaccSplitGetNoclosingBalance);
3516 : : }
3517 : :
3518 : : gnc_numeric
3519 : 0 : xaccAccountGetReconciledBalanceAsOfDate (Account *acc, time64 date)
3520 : : {
3521 : 0 : return GetBalanceAsOfDate (acc, date, xaccSplitGetReconciledBalance);
3522 : : }
3523 : :
3524 : : /*
3525 : : * Originally gsr_account_present_balance in gnc-split-reg.c
3526 : : */
3527 : : gnc_numeric
3528 : 1 : xaccAccountGetPresentBalance (const Account *acc)
3529 : : {
3530 : 1 : g_return_val_if_fail(GNC_IS_ACCOUNT(acc), gnc_numeric_zero());
3531 : :
3532 : 1 : return xaccAccountGetBalanceAsOfDate (GNC_ACCOUNT (acc),
3533 : 1 : gnc_time64_get_today_end ());
3534 : : }
3535 : :
3536 : :
3537 : : /********************************************************************\
3538 : : \********************************************************************/
3539 : : /* XXX TODO: These 'GetBal' routines should be moved to some
3540 : : * utility area outside of the core account engine area.
3541 : : */
3542 : :
3543 : : /*
3544 : : * Convert a balance from one currency to another.
3545 : : */
3546 : : gnc_numeric
3547 : 0 : xaccAccountConvertBalanceToCurrency(const Account *acc, /* for book */
3548 : : gnc_numeric balance,
3549 : : const gnc_commodity *balance_currency,
3550 : : const gnc_commodity *new_currency)
3551 : : {
3552 : : QofBook *book;
3553 : : GNCPriceDB *pdb;
3554 : :
3555 : 0 : if (gnc_numeric_zero_p (balance) ||
3556 : 0 : gnc_commodity_equiv (balance_currency, new_currency))
3557 : 0 : return balance;
3558 : :
3559 : 0 : book = gnc_account_get_book (acc);
3560 : 0 : pdb = gnc_pricedb_get_db (book);
3561 : :
3562 : 0 : balance = gnc_pricedb_convert_balance_latest_price(
3563 : : pdb, balance, balance_currency, new_currency);
3564 : :
3565 : 0 : return balance;
3566 : : }
3567 : :
3568 : : /*
3569 : : * Convert a balance from one currency to another with price of
3570 : : * a given date.
3571 : : */
3572 : : gnc_numeric
3573 : 732 : xaccAccountConvertBalanceToCurrencyAsOfDate(const Account *acc, /* for book */
3574 : : gnc_numeric balance,
3575 : : const gnc_commodity *balance_currency,
3576 : : const gnc_commodity *new_currency,
3577 : : time64 date)
3578 : : {
3579 : : QofBook *book;
3580 : : GNCPriceDB *pdb;
3581 : :
3582 : 884 : if (gnc_numeric_zero_p (balance) ||
3583 : 152 : gnc_commodity_equiv (balance_currency, new_currency))
3584 : 732 : return balance;
3585 : :
3586 : 0 : book = gnc_account_get_book (acc);
3587 : 0 : pdb = gnc_pricedb_get_db (book);
3588 : :
3589 : 0 : balance = gnc_pricedb_convert_balance_nearest_before_price_t64 (
3590 : : pdb, balance, balance_currency, new_currency, date);
3591 : :
3592 : 0 : return balance;
3593 : : }
3594 : :
3595 : : /*
3596 : : * Given an account and a GetBalanceFn pointer, extract the requested
3597 : : * balance from the account and then convert it to the desired
3598 : : * currency.
3599 : : */
3600 : : static gnc_numeric
3601 : 0 : xaccAccountGetXxxBalanceInCurrency (const Account *acc,
3602 : : xaccGetBalanceFn fn,
3603 : : const gnc_commodity *report_currency)
3604 : : {
3605 : : AccountPrivate *priv;
3606 : : gnc_numeric balance;
3607 : :
3608 : 0 : g_return_val_if_fail(GNC_IS_ACCOUNT(acc), gnc_numeric_zero());
3609 : 0 : g_return_val_if_fail(fn, gnc_numeric_zero());
3610 : 0 : g_return_val_if_fail(GNC_IS_COMMODITY(report_currency), gnc_numeric_zero());
3611 : :
3612 : 0 : priv = GET_PRIVATE(acc);
3613 : 0 : balance = fn(acc);
3614 : 0 : balance = xaccAccountConvertBalanceToCurrency(acc, balance,
3615 : 0 : priv->commodity,
3616 : : report_currency);
3617 : 0 : return balance;
3618 : : }
3619 : :
3620 : : static gnc_numeric
3621 : 0 : xaccAccountGetXxxBalanceAsOfDateInCurrency(Account *acc, time64 date,
3622 : : xaccGetBalanceAsOfDateFn fn,
3623 : : const gnc_commodity *report_commodity)
3624 : : {
3625 : : AccountPrivate *priv;
3626 : :
3627 : 0 : g_return_val_if_fail(GNC_IS_ACCOUNT(acc), gnc_numeric_zero());
3628 : 0 : g_return_val_if_fail(fn, gnc_numeric_zero());
3629 : 0 : g_return_val_if_fail(GNC_IS_COMMODITY(report_commodity), gnc_numeric_zero());
3630 : :
3631 : 0 : priv = GET_PRIVATE(acc);
3632 : 0 : return xaccAccountConvertBalanceToCurrencyAsOfDate(
3633 : 0 : acc, fn(acc, date), priv->commodity, report_commodity, date);
3634 : : }
3635 : :
3636 : : /*
3637 : : * Data structure used to pass various arguments into the following fn.
3638 : : */
3639 : : typedef struct
3640 : : {
3641 : : const gnc_commodity *currency;
3642 : : gnc_numeric balance;
3643 : : xaccGetBalanceFn fn;
3644 : : xaccGetBalanceAsOfDateFn asOfDateFn;
3645 : : time64 date;
3646 : : } CurrencyBalance;
3647 : :
3648 : :
3649 : : /*
3650 : : * A helper function for iterating over all the accounts in a list or
3651 : : * tree. This function is called once per account, and sums up the
3652 : : * values of all these accounts.
3653 : : */
3654 : : static void
3655 : 0 : xaccAccountBalanceHelper (Account *acc, gpointer data)
3656 : : {
3657 : 0 : CurrencyBalance *cb = static_cast<CurrencyBalance*>(data);
3658 : : gnc_numeric balance;
3659 : :
3660 : 0 : if (!cb->fn || !cb->currency)
3661 : 0 : return;
3662 : 0 : balance = xaccAccountGetXxxBalanceInCurrency (acc, cb->fn, cb->currency);
3663 : 0 : cb->balance = gnc_numeric_add (cb->balance, balance,
3664 : 0 : gnc_commodity_get_fraction (cb->currency),
3665 : : GNC_HOW_RND_ROUND_HALF_UP);
3666 : : }
3667 : :
3668 : : static void
3669 : 0 : xaccAccountBalanceAsOfDateHelper (Account *acc, gpointer data)
3670 : : {
3671 : 0 : CurrencyBalance *cb = static_cast<CurrencyBalance*>(data);
3672 : : gnc_numeric balance;
3673 : :
3674 : 0 : g_return_if_fail (cb->asOfDateFn && cb->currency);
3675 : :
3676 : 0 : balance = xaccAccountGetXxxBalanceAsOfDateInCurrency (
3677 : : acc, cb->date, cb->asOfDateFn, cb->currency);
3678 : 0 : cb->balance = gnc_numeric_add (cb->balance, balance,
3679 : 0 : gnc_commodity_get_fraction (cb->currency),
3680 : : GNC_HOW_RND_ROUND_HALF_UP);
3681 : : }
3682 : :
3683 : :
3684 : :
3685 : : /*
3686 : : * Common function that iterates recursively over all accounts below
3687 : : * the specified account. It uses xaccAccountBalanceHelper to sum up
3688 : : * the balances of all its children, and uses the specified function
3689 : : * 'fn' for extracting the balance. This function may extract the
3690 : : * current value, the reconciled value, etc.
3691 : : *
3692 : : * If 'report_commodity' is nullptr, just use the account's commodity.
3693 : : * If 'include_children' is FALSE, this function doesn't recurse at all.
3694 : : */
3695 : : static gnc_numeric
3696 : 0 : xaccAccountGetXxxBalanceInCurrencyRecursive (const Account *acc,
3697 : : xaccGetBalanceFn fn,
3698 : : const gnc_commodity *report_commodity,
3699 : : gboolean include_children)
3700 : : {
3701 : : gnc_numeric balance;
3702 : :
3703 : 0 : if (!acc) return gnc_numeric_zero ();
3704 : 0 : if (!report_commodity)
3705 : 0 : report_commodity = xaccAccountGetCommodity (acc);
3706 : 0 : if (!report_commodity)
3707 : 0 : return gnc_numeric_zero();
3708 : :
3709 : 0 : balance = xaccAccountGetXxxBalanceInCurrency (acc, fn, report_commodity);
3710 : :
3711 : : /* If needed, sum up the children converting to the *requested*
3712 : : commodity. */
3713 : 0 : if (include_children)
3714 : : {
3715 : : #ifdef _MSC_VER
3716 : : /* MSVC compiler: Somehow, the struct initialization containing a
3717 : : gnc_numeric doesn't work. As an exception, we hand-initialize
3718 : : that member afterwards. */
3719 : : CurrencyBalance cb = { report_commodity, { 0 }, fn, nullptr, 0 };
3720 : : cb.balance = balance;
3721 : : #else
3722 : 0 : CurrencyBalance cb = { report_commodity, balance, fn, nullptr, 0 };
3723 : : #endif
3724 : :
3725 : 0 : gnc_account_foreach_descendant (acc, xaccAccountBalanceHelper, &cb);
3726 : 0 : balance = cb.balance;
3727 : : }
3728 : :
3729 : 0 : return balance;
3730 : : }
3731 : :
3732 : : static gnc_numeric
3733 : 0 : xaccAccountGetXxxBalanceAsOfDateInCurrencyRecursive (
3734 : : Account *acc, time64 date, xaccGetBalanceAsOfDateFn fn,
3735 : : const gnc_commodity *report_commodity, gboolean include_children)
3736 : : {
3737 : : gnc_numeric balance;
3738 : :
3739 : 0 : g_return_val_if_fail(acc, gnc_numeric_zero());
3740 : 0 : if (!report_commodity)
3741 : 0 : report_commodity = xaccAccountGetCommodity (acc);
3742 : 0 : if (!report_commodity)
3743 : 0 : return gnc_numeric_zero();
3744 : :
3745 : 0 : balance = xaccAccountGetXxxBalanceAsOfDateInCurrency(
3746 : : acc, date, fn, report_commodity);
3747 : :
3748 : : /* If needed, sum up the children converting to the *requested*
3749 : : commodity. */
3750 : 0 : if (include_children)
3751 : : {
3752 : : #ifdef _MSC_VER
3753 : : /* MSVC compiler: Somehow, the struct initialization containing a
3754 : : gnc_numeric doesn't work. As an exception, we hand-initialize
3755 : : that member afterwards. */
3756 : : CurrencyBalance cb = { report_commodity, 0, nullptr, fn, date };
3757 : : cb.balance = balance;
3758 : : #else
3759 : 0 : CurrencyBalance cb = { report_commodity, balance, nullptr, fn, date };
3760 : : #endif
3761 : :
3762 : 0 : gnc_account_foreach_descendant (acc, xaccAccountBalanceAsOfDateHelper, &cb);
3763 : 0 : balance = cb.balance;
3764 : : }
3765 : :
3766 : 0 : return balance;
3767 : : }
3768 : :
3769 : : gnc_numeric
3770 : 0 : xaccAccountGetBalanceInCurrency (const Account *acc,
3771 : : const gnc_commodity *report_commodity,
3772 : : gboolean include_children)
3773 : : {
3774 : : gnc_numeric rc;
3775 : 0 : rc = xaccAccountGetXxxBalanceInCurrencyRecursive (
3776 : : acc, xaccAccountGetBalance, report_commodity, include_children);
3777 : 0 : PINFO(" baln=%" G_GINT64_FORMAT "/%" G_GINT64_FORMAT, rc.num, rc.denom);
3778 : 0 : return rc;
3779 : : }
3780 : :
3781 : :
3782 : : gnc_numeric
3783 : 0 : xaccAccountGetClearedBalanceInCurrency (const Account *acc,
3784 : : const gnc_commodity *report_commodity,
3785 : : gboolean include_children)
3786 : : {
3787 : 0 : return xaccAccountGetXxxBalanceInCurrencyRecursive (
3788 : : acc, xaccAccountGetClearedBalance, report_commodity,
3789 : 0 : include_children);
3790 : : }
3791 : :
3792 : : gnc_numeric
3793 : 0 : xaccAccountGetReconciledBalanceInCurrency (const Account *acc,
3794 : : const gnc_commodity *report_commodity,
3795 : : gboolean include_children)
3796 : : {
3797 : 0 : return xaccAccountGetXxxBalanceInCurrencyRecursive (
3798 : : acc, xaccAccountGetReconciledBalance, report_commodity,
3799 : 0 : include_children);
3800 : : }
3801 : :
3802 : : gnc_numeric
3803 : 0 : xaccAccountGetPresentBalanceInCurrency (const Account *acc,
3804 : : const gnc_commodity *report_commodity,
3805 : : gboolean include_children)
3806 : : {
3807 : 0 : return xaccAccountGetXxxBalanceAsOfDateInCurrencyRecursive (
3808 : : (Account*)acc, gnc_time64_get_today_end (), xaccAccountGetBalanceAsOfDate,
3809 : : report_commodity,
3810 : 0 : include_children);
3811 : : }
3812 : :
3813 : : gnc_numeric
3814 : 0 : xaccAccountGetProjectedMinimumBalanceInCurrency (
3815 : : const Account *acc,
3816 : : const gnc_commodity *report_commodity,
3817 : : gboolean include_children)
3818 : : {
3819 : 0 : return xaccAccountGetXxxBalanceInCurrencyRecursive (
3820 : : acc, xaccAccountGetProjectedMinimumBalance, report_commodity,
3821 : 0 : include_children);
3822 : : }
3823 : :
3824 : : gnc_numeric
3825 : 0 : xaccAccountGetBalanceAsOfDateInCurrency(
3826 : : Account *acc, time64 date, gnc_commodity *report_commodity,
3827 : : gboolean include_children)
3828 : : {
3829 : 0 : return xaccAccountGetXxxBalanceAsOfDateInCurrencyRecursive (
3830 : : acc, date, xaccAccountGetBalanceAsOfDate, report_commodity,
3831 : 0 : include_children);
3832 : : }
3833 : :
3834 : : gnc_numeric
3835 : 0 : xaccAccountGetNoclosingBalanceAsOfDateInCurrency(
3836 : : Account *acc, time64 date, gnc_commodity *report_commodity,
3837 : : gboolean include_children)
3838 : : {
3839 : : return xaccAccountGetXxxBalanceAsOfDateInCurrencyRecursive
3840 : 0 : (acc, date, xaccAccountGetNoclosingBalanceAsOfDate,
3841 : 0 : report_commodity, include_children);
3842 : : }
3843 : :
3844 : : gnc_numeric
3845 : 0 : xaccAccountGetBalanceChangeForPeriod (Account *acc, time64 t1, time64 t2,
3846 : : gboolean recurse)
3847 : : {
3848 : : gnc_numeric b1, b2;
3849 : :
3850 : 0 : b1 = xaccAccountGetBalanceAsOfDateInCurrency(acc, t1, nullptr, recurse);
3851 : 0 : b2 = xaccAccountGetBalanceAsOfDateInCurrency(acc, t2, nullptr, recurse);
3852 : 0 : return gnc_numeric_sub(b2, b1, GNC_DENOM_AUTO, GNC_HOW_DENOM_FIXED);
3853 : : }
3854 : :
3855 : : gnc_numeric
3856 : 0 : xaccAccountGetNoclosingBalanceChangeForPeriod (Account *acc, time64 t1,
3857 : : time64 t2, gboolean recurse)
3858 : : {
3859 : : gnc_numeric b1, b2;
3860 : :
3861 : 0 : b1 = xaccAccountGetNoclosingBalanceAsOfDateInCurrency(acc, t1, nullptr, recurse);
3862 : 0 : b2 = xaccAccountGetNoclosingBalanceAsOfDateInCurrency(acc, t2, nullptr, recurse);
3863 : 0 : return gnc_numeric_sub(b2, b1, GNC_DENOM_AUTO, GNC_HOW_DENOM_FIXED);
3864 : : }
3865 : :
3866 : : typedef struct
3867 : : {
3868 : : const gnc_commodity *currency;
3869 : : gnc_numeric balanceChange;
3870 : : time64 t1;
3871 : : time64 t2;
3872 : : } CurrencyBalanceChange;
3873 : :
3874 : : static void
3875 : 732 : xaccAccountBalanceChangeHelper (Account *acc, gpointer data)
3876 : : {
3877 : 732 : CurrencyBalanceChange *cbdiff = static_cast<CurrencyBalanceChange*>(data);
3878 : :
3879 : : gnc_numeric b1, b2;
3880 : 732 : b1 = GetBalanceAsOfDate(acc, cbdiff->t1, xaccSplitGetNoclosingBalance);
3881 : 732 : b2 = GetBalanceAsOfDate(acc, cbdiff->t2, xaccSplitGetNoclosingBalance);
3882 : 732 : gnc_numeric balanceChange = gnc_numeric_sub(b2, b1, GNC_DENOM_AUTO, GNC_HOW_DENOM_FIXED);
3883 : 732 : gnc_numeric balanceChange_conv = xaccAccountConvertBalanceToCurrencyAsOfDate(acc, balanceChange, xaccAccountGetCommodity(acc), cbdiff->currency, cbdiff->t2);
3884 : 732 : cbdiff->balanceChange = gnc_numeric_add (cbdiff->balanceChange, balanceChange_conv,
3885 : 732 : gnc_commodity_get_fraction (cbdiff->currency),
3886 : : GNC_HOW_RND_ROUND_HALF_UP);
3887 : 732 : }
3888 : :
3889 : : gnc_numeric
3890 : 549 : xaccAccountGetNoclosingBalanceChangeInCurrencyForPeriod (Account *acc, time64 t1,
3891 : : time64 t2, gboolean recurse)
3892 : : {
3893 : :
3894 : :
3895 : : gnc_numeric b1, b2;
3896 : 549 : b1 = GetBalanceAsOfDate(acc, t1, xaccSplitGetNoclosingBalance);
3897 : 549 : b2 = GetBalanceAsOfDate(acc, t2, xaccSplitGetNoclosingBalance);
3898 : 549 : gnc_numeric balanceChange = gnc_numeric_sub(b2, b1, GNC_DENOM_AUTO, GNC_HOW_DENOM_FIXED);
3899 : :
3900 : 549 : gnc_commodity *report_commodity = xaccAccountGetCommodity(acc);
3901 : 549 : CurrencyBalanceChange cbdiff = { report_commodity, balanceChange, t1, t2 };
3902 : :
3903 : 549 : if(recurse)
3904 : : {
3905 : 549 : gnc_account_foreach_descendant (acc, xaccAccountBalanceChangeHelper, &cbdiff);
3906 : 549 : balanceChange = cbdiff.balanceChange;
3907 : : }
3908 : 549 : return balanceChange;
3909 : : }
3910 : :
3911 : : /********************************************************************\
3912 : : \********************************************************************/
3913 : :
3914 : : const SplitsVec&
3915 : 939 : xaccAccountGetSplits (const Account *account)
3916 : : {
3917 : 939 : static const SplitsVec empty;
3918 : 939 : g_return_val_if_fail (GNC_IS_ACCOUNT(account), empty);
3919 : 939 : return GET_PRIVATE(account)->splits;
3920 : : }
3921 : :
3922 : : SplitList *
3923 : 1 : xaccAccountGetSplitList (const Account *acc)
3924 : : {
3925 : 1 : g_return_val_if_fail(GNC_IS_ACCOUNT(acc), nullptr);
3926 : 1 : auto priv{GET_PRIVATE(acc)};
3927 : 2 : return std::accumulate (priv->splits.rbegin(), priv->splits.rend(),
3928 : 1 : static_cast<GList*>(nullptr), g_list_prepend);
3929 : : }
3930 : :
3931 : : size_t
3932 : 23 : xaccAccountGetSplitsSize (const Account *account)
3933 : : {
3934 : 23 : g_return_val_if_fail (GNC_IS_ACCOUNT(account), 0);
3935 : 23 : return GNC_IS_ACCOUNT(account) ? GET_PRIVATE(account)->splits.size() : 0;
3936 : : }
3937 : :
3938 : 0 : gboolean gnc_account_and_descendants_empty (Account *acc)
3939 : : {
3940 : 0 : g_return_val_if_fail (GNC_IS_ACCOUNT (acc), FALSE);
3941 : 0 : auto priv = GET_PRIVATE (acc);
3942 : 0 : if (!priv->splits.empty()) return FALSE;
3943 : 0 : return std::all_of (priv->children.begin(), priv->children.end(),
3944 : 0 : gnc_account_and_descendants_empty);
3945 : : }
3946 : :
3947 : : LotList *
3948 : 468 : xaccAccountGetLotList (const Account *acc)
3949 : : {
3950 : 468 : g_return_val_if_fail(GNC_IS_ACCOUNT(acc), nullptr);
3951 : 468 : return g_list_copy(GET_PRIVATE(acc)->lots);
3952 : : }
3953 : :
3954 : : LotList *
3955 : 5 : xaccAccountFindOpenLots (const Account *acc,
3956 : : gboolean (*match_func)(GNCLot *lot,
3957 : : gpointer user_data),
3958 : : gpointer user_data, GCompareFunc sort_func)
3959 : : {
3960 : : AccountPrivate *priv;
3961 : : GList *lot_list;
3962 : 5 : GList *retval = nullptr;
3963 : :
3964 : 5 : g_return_val_if_fail(GNC_IS_ACCOUNT(acc), nullptr);
3965 : :
3966 : 5 : priv = GET_PRIVATE(acc);
3967 : 18 : for (lot_list = priv->lots; lot_list; lot_list = lot_list->next)
3968 : : {
3969 : 13 : GNCLot *lot = static_cast<GNCLot*>(lot_list->data);
3970 : :
3971 : : /* If this lot is closed, then ignore it */
3972 : 13 : if (gnc_lot_is_closed (lot))
3973 : 5 : continue;
3974 : :
3975 : 8 : if (match_func && !(match_func)(lot, user_data))
3976 : 2 : continue;
3977 : :
3978 : : /* Ok, this is a valid lot. Add it to our list of lots */
3979 : 6 : retval = g_list_prepend (retval, lot);
3980 : : }
3981 : :
3982 : 5 : if (sort_func)
3983 : 1 : retval = g_list_sort (retval, sort_func);
3984 : :
3985 : 5 : return retval;
3986 : : }
3987 : :
3988 : : gpointer
3989 : 66 : xaccAccountForEachLot(const Account *acc,
3990 : : gpointer (*proc)(GNCLot *lot, void *data), void *data)
3991 : : {
3992 : 66 : g_return_val_if_fail(GNC_IS_ACCOUNT(acc), nullptr);
3993 : 66 : g_return_val_if_fail(proc, nullptr);
3994 : :
3995 : 827 : for (auto node = GET_PRIVATE(acc)->lots; node; node = node->next)
3996 : 762 : if (auto result = proc(GNC_LOT(node->data), data))
3997 : 1 : return result;
3998 : :
3999 : 65 : return nullptr;
4000 : : }
4001 : :
4002 : :
4003 : : /********************************************************************\
4004 : : \********************************************************************/
4005 : :
4006 : : /* These functions use interchange gint64 and gboolean. Is that right? */
4007 : : gboolean
4008 : 1 : xaccAccountGetTaxRelated (const Account *acc)
4009 : : {
4010 : 3 : return get_kvp_boolean_path(acc, {"tax-related"});
4011 : : }
4012 : :
4013 : : void
4014 : 0 : xaccAccountSetTaxRelated (Account *acc, gboolean tax_related)
4015 : : {
4016 : 0 : set_kvp_boolean_path(acc, {"tax-related"}, tax_related);
4017 : 0 : }
4018 : :
4019 : : const char *
4020 : 7 : xaccAccountGetTaxUSCode (const Account *acc)
4021 : : {
4022 : 21 : return get_kvp_string_path (acc, {"tax-US", "code"});
4023 : : }
4024 : :
4025 : : void
4026 : 5 : xaccAccountSetTaxUSCode (Account *acc, const char *code)
4027 : : {
4028 : 10 : set_kvp_string_path (acc, {"tax-US", "code"}, code);
4029 : 5 : }
4030 : :
4031 : : const char *
4032 : 7 : xaccAccountGetTaxUSPayerNameSource (const Account *acc)
4033 : : {
4034 : 21 : return get_kvp_string_path (acc, {"tax-US", "payer-name-source"});
4035 : : }
4036 : :
4037 : : void
4038 : 5 : xaccAccountSetTaxUSPayerNameSource (Account *acc, const char *source)
4039 : : {
4040 : 10 : set_kvp_string_path (acc, {"tax-US", "payer-name-source"}, source);
4041 : 5 : }
4042 : :
4043 : : gint64
4044 : 1 : xaccAccountGetTaxUSCopyNumber (const Account *acc)
4045 : : {
4046 : 2 : auto copy_number = get_kvp_int64_path (acc, {"tax-US", "copy-number"});
4047 : 2 : return (copy_number && (*copy_number != 0)) ? *copy_number : 1;
4048 : : }
4049 : :
4050 : : void
4051 : 0 : xaccAccountSetTaxUSCopyNumber (Account *acc, gint64 copy_number)
4052 : : {
4053 : 0 : if (copy_number != 0)
4054 : 0 : set_kvp_int64_path (acc, {"tax-US", "copy-number"}, copy_number);
4055 : : else
4056 : : /* deletes KVP if it exists */
4057 : 0 : set_kvp_int64_path (acc, {"tax-US", "copy-number"}, std::nullopt);
4058 : 0 : }
4059 : :
4060 : : /*********************************************************************\
4061 : : \ ********************************************************************/
4062 : :
4063 : :
4064 : 11 : const char *gnc_account_get_debit_string (GNCAccountType acct_type)
4065 : : {
4066 : 11 : if (gnc_prefs_get_bool(GNC_PREFS_GROUP_GENERAL, GNC_PREF_ACCOUNTING_LABELS))
4067 : 0 : return _(dflt_acct_debit_str);
4068 : :
4069 : 11 : auto result = gnc_acct_debit_strs.find(acct_type);
4070 : 11 : if (result != gnc_acct_debit_strs.end())
4071 : 11 : return _(result->second);
4072 : : else
4073 : 0 : return _(dflt_acct_debit_str);
4074 : : }
4075 : :
4076 : 11 : const char *gnc_account_get_credit_string (GNCAccountType acct_type)
4077 : : {
4078 : 11 : if (gnc_prefs_get_bool(GNC_PREFS_GROUP_GENERAL, GNC_PREF_ACCOUNTING_LABELS))
4079 : 0 : return _(dflt_acct_credit_str);
4080 : :
4081 : 11 : auto result = gnc_acct_credit_strs.find(acct_type);
4082 : 11 : if (result != gnc_acct_credit_strs.end())
4083 : 11 : return _(result->second);
4084 : : else
4085 : 0 : return _(dflt_acct_credit_str);
4086 : : }
4087 : :
4088 : : /********************************************************************\
4089 : : \********************************************************************/
4090 : :
4091 : : gboolean
4092 : 209 : xaccAccountGetPlaceholder (const Account *acc)
4093 : : {
4094 : 627 : return get_kvp_boolean_path(acc, {"placeholder"});
4095 : : }
4096 : :
4097 : : void
4098 : 126 : xaccAccountSetPlaceholder (Account *acc, gboolean val)
4099 : : {
4100 : 252 : set_kvp_boolean_path(acc, {"placeholder"}, val);
4101 : 126 : }
4102 : :
4103 : : gboolean
4104 : 0 : xaccAccountGetAppendText (const Account *acc)
4105 : : {
4106 : 0 : return get_kvp_boolean_path(acc, {"import-append-text"});
4107 : : }
4108 : :
4109 : : void
4110 : 0 : xaccAccountSetAppendText (Account *acc, gboolean val)
4111 : : {
4112 : 0 : set_kvp_boolean_path(acc, {"import-append-text"}, val);
4113 : 0 : }
4114 : :
4115 : : gboolean
4116 : 4 : xaccAccountGetIsOpeningBalance (const Account *acc)
4117 : : {
4118 : 4 : g_return_val_if_fail (GNC_IS_ACCOUNT(acc), false);
4119 : 4 : if (GET_PRIVATE(acc)->type != ACCT_TYPE_EQUITY)
4120 : 1 : return false;
4121 : :
4122 : 9 : return !g_strcmp0 (get_kvp_string_path (acc, {"equity-type"}), "opening-balance");
4123 : : }
4124 : :
4125 : : void
4126 : 2 : xaccAccountSetIsOpeningBalance (Account *acc, gboolean val)
4127 : : {
4128 : 2 : g_return_if_fail (GNC_IS_ACCOUNT(acc));
4129 : 2 : if (GET_PRIVATE(acc)->type != ACCT_TYPE_EQUITY)
4130 : 0 : return;
4131 : 6 : set_kvp_string_path(acc, {"equity-type"}, val ? "opening-balance" : nullptr);
4132 : : }
4133 : :
4134 : : GNCPlaceholderType
4135 : 0 : xaccAccountGetDescendantPlaceholder (const Account *acc)
4136 : : {
4137 : 0 : g_return_val_if_fail(GNC_IS_ACCOUNT(acc), PLACEHOLDER_NONE);
4138 : 0 : if (xaccAccountGetPlaceholder(acc)) return PLACEHOLDER_THIS;
4139 : :
4140 : 0 : return gnc_account_foreach_descendant_until (acc, (AccountCb2)xaccAccountGetPlaceholder, nullptr)
4141 : 0 : ? PLACEHOLDER_CHILD : PLACEHOLDER_NONE;
4142 : : }
4143 : :
4144 : : /********************************************************************\
4145 : : \********************************************************************/
4146 : :
4147 : : gboolean
4148 : 0 : xaccAccountGetAutoInterest (const Account *acc)
4149 : : {
4150 : 0 : return get_kvp_boolean_path (acc, {KEY_RECONCILE_INFO, "auto-interest-transfer"});
4151 : 0 : }
4152 : :
4153 : : void
4154 : 0 : xaccAccountSetAutoInterest (Account *acc, gboolean val)
4155 : : {
4156 : 0 : set_kvp_boolean_path (acc, {KEY_RECONCILE_INFO, "auto-interest-transfer"}, val);
4157 : 0 : }
4158 : :
4159 : : /********************************************************************\
4160 : : \********************************************************************/
4161 : :
4162 : : gboolean
4163 : 43 : xaccAccountGetHidden (const Account *acc)
4164 : : {
4165 : 129 : return get_kvp_boolean_path (acc, {"hidden"});
4166 : : }
4167 : :
4168 : : void
4169 : 46 : xaccAccountSetHidden (Account *acc, gboolean val)
4170 : : {
4171 : 92 : set_kvp_boolean_path (acc, {"hidden"}, val);
4172 : 46 : }
4173 : :
4174 : : gboolean
4175 : 0 : xaccAccountIsHidden (const Account *acc)
4176 : : {
4177 : : AccountPrivate *priv;
4178 : :
4179 : 0 : g_return_val_if_fail(GNC_IS_ACCOUNT(acc), FALSE);
4180 : :
4181 : 0 : if (xaccAccountGetHidden(acc))
4182 : 0 : return TRUE;
4183 : 0 : priv = GET_PRIVATE(acc);
4184 : 0 : while ((acc = priv->parent) != nullptr)
4185 : : {
4186 : 0 : priv = GET_PRIVATE(acc);
4187 : 0 : if (xaccAccountGetHidden(acc))
4188 : 0 : return TRUE;
4189 : : }
4190 : 0 : return FALSE;
4191 : : }
4192 : :
4193 : : /********************************************************************\
4194 : : \********************************************************************/
4195 : :
4196 : : gboolean
4197 : 3 : xaccAccountHasAncestor (const Account *acc, const Account * ancestor)
4198 : : {
4199 : : const Account *parent;
4200 : :
4201 : 3 : g_return_val_if_fail(GNC_IS_ACCOUNT(acc), FALSE);
4202 : 3 : g_return_val_if_fail(GNC_IS_ACCOUNT(ancestor), FALSE);
4203 : :
4204 : 3 : parent = acc;
4205 : 13 : while (parent && parent != ancestor)
4206 : 10 : parent = GET_PRIVATE(parent)->parent;
4207 : :
4208 : 3 : return (parent == ancestor);
4209 : : }
4210 : :
4211 : : /********************************************************************\
4212 : : \********************************************************************/
4213 : :
4214 : : /* You must edit the functions in this block in tandem. KEEP THEM IN
4215 : : SYNC! */
4216 : :
4217 : : #define GNC_RETURN_ENUM_AS_STRING(x) case (ACCT_TYPE_ ## x): return #x;
4218 : :
4219 : : const char *
4220 : 314 : xaccAccountTypeEnumAsString(GNCAccountType type)
4221 : : {
4222 : 314 : switch (type)
4223 : : {
4224 : 1 : GNC_RETURN_ENUM_AS_STRING(NONE);
4225 : 26 : GNC_RETURN_ENUM_AS_STRING(BANK);
4226 : 6 : GNC_RETURN_ENUM_AS_STRING(CASH);
4227 : 4 : GNC_RETURN_ENUM_AS_STRING(CREDIT);
4228 : 13 : GNC_RETURN_ENUM_AS_STRING(ASSET);
4229 : 16 : GNC_RETURN_ENUM_AS_STRING(LIABILITY);
4230 : 2 : GNC_RETURN_ENUM_AS_STRING(STOCK);
4231 : 4 : GNC_RETURN_ENUM_AS_STRING(MUTUAL);
4232 : 4 : GNC_RETURN_ENUM_AS_STRING(CURRENCY);
4233 : 26 : GNC_RETURN_ENUM_AS_STRING(INCOME);
4234 : 164 : GNC_RETURN_ENUM_AS_STRING(EXPENSE);
4235 : 11 : GNC_RETURN_ENUM_AS_STRING(EQUITY);
4236 : 4 : GNC_RETURN_ENUM_AS_STRING(RECEIVABLE);
4237 : 3 : GNC_RETURN_ENUM_AS_STRING(PAYABLE);
4238 : 24 : GNC_RETURN_ENUM_AS_STRING(ROOT);
4239 : 1 : GNC_RETURN_ENUM_AS_STRING(TRADING);
4240 : 1 : GNC_RETURN_ENUM_AS_STRING(CHECKING);
4241 : 1 : GNC_RETURN_ENUM_AS_STRING(SAVINGS);
4242 : 1 : GNC_RETURN_ENUM_AS_STRING(MONEYMRKT);
4243 : 1 : GNC_RETURN_ENUM_AS_STRING(CREDITLINE);
4244 : 1 : default:
4245 : 1 : PERR ("asked to translate unknown account type %d.\n", type);
4246 : 1 : break;
4247 : : }
4248 : 1 : return(nullptr);
4249 : : }
4250 : :
4251 : : #undef GNC_RETURN_ENUM_AS_STRING
4252 : :
4253 : : #define GNC_RETURN_ON_MATCH(x) \
4254 : : if(g_strcmp0(#x, (str)) == 0) { *type = ACCT_TYPE_ ## x; return(TRUE); }
4255 : :
4256 : : gboolean
4257 : 1293 : xaccAccountStringToType(const char* str, GNCAccountType *type)
4258 : : {
4259 : :
4260 : 1293 : GNC_RETURN_ON_MATCH(NONE);
4261 : 1292 : GNC_RETURN_ON_MATCH(BANK);
4262 : 1202 : GNC_RETURN_ON_MATCH(CASH);
4263 : 1186 : GNC_RETURN_ON_MATCH(CREDIT);
4264 : 1169 : GNC_RETURN_ON_MATCH(ASSET);
4265 : 1091 : GNC_RETURN_ON_MATCH(LIABILITY);
4266 : 1027 : GNC_RETURN_ON_MATCH(STOCK);
4267 : 989 : GNC_RETURN_ON_MATCH(MUTUAL);
4268 : 962 : GNC_RETURN_ON_MATCH(CURRENCY);
4269 : 953 : GNC_RETURN_ON_MATCH(INCOME);
4270 : 806 : GNC_RETURN_ON_MATCH(EXPENSE);
4271 : 141 : GNC_RETURN_ON_MATCH(EQUITY);
4272 : 99 : GNC_RETURN_ON_MATCH(RECEIVABLE);
4273 : 91 : GNC_RETURN_ON_MATCH(PAYABLE);
4274 : 86 : GNC_RETURN_ON_MATCH(ROOT);
4275 : 7 : GNC_RETURN_ON_MATCH(TRADING);
4276 : 6 : GNC_RETURN_ON_MATCH(CHECKING);
4277 : 5 : GNC_RETURN_ON_MATCH(SAVINGS);
4278 : 4 : GNC_RETURN_ON_MATCH(MONEYMRKT);
4279 : 3 : GNC_RETURN_ON_MATCH(CREDITLINE);
4280 : :
4281 : 2 : PERR("asked to translate unknown account type string %s.\n",
4282 : : str ? str : "(null)");
4283 : :
4284 : 2 : return(FALSE);
4285 : : }
4286 : :
4287 : : #undef GNC_RETURN_ON_MATCH
4288 : :
4289 : : /* impedance mismatch is a source of loss */
4290 : : GNCAccountType
4291 : 62 : xaccAccountStringToEnum(const char* str)
4292 : : {
4293 : : GNCAccountType type;
4294 : : gboolean rc;
4295 : 62 : rc = xaccAccountStringToType(str, &type);
4296 : 62 : if (FALSE == rc) return ACCT_TYPE_INVALID;
4297 : 62 : return type;
4298 : : }
4299 : :
4300 : : /********************************************************************\
4301 : : \********************************************************************/
4302 : :
4303 : : static char const *
4304 : : account_type_name[NUM_ACCOUNT_TYPES] =
4305 : : {
4306 : : N_("Bank"),
4307 : : N_("Cash"),
4308 : : N_("Asset"),
4309 : : N_("Credit Card"),
4310 : : N_("Liability"),
4311 : : N_("Stock"),
4312 : : N_("Mutual Fund"),
4313 : : N_("Currency"),
4314 : : N_("Income"),
4315 : : N_("Expense"),
4316 : : N_("Equity"),
4317 : : N_("A/Receivable"),
4318 : : N_("A/Payable"),
4319 : : N_("Root"),
4320 : : N_("Trading")
4321 : : /*
4322 : : N_("Checking"),
4323 : : N_("Savings"),
4324 : : N_("Money Market"),
4325 : : N_("Credit Line")
4326 : : */
4327 : : };
4328 : :
4329 : : const char *
4330 : 1399 : xaccAccountGetTypeStr(GNCAccountType type)
4331 : : {
4332 : 1399 : if (type < 0 || NUM_ACCOUNT_TYPES <= type ) return "";
4333 : 1399 : return _(account_type_name [type]);
4334 : : }
4335 : :
4336 : : /********************************************************************\
4337 : : \********************************************************************/
4338 : :
4339 : : guint32
4340 : 14 : xaccAccountTypesCompatibleWith (GNCAccountType type)
4341 : : {
4342 : 14 : switch (type)
4343 : : {
4344 : 8 : case ACCT_TYPE_BANK:
4345 : : case ACCT_TYPE_CASH:
4346 : : case ACCT_TYPE_ASSET:
4347 : : case ACCT_TYPE_CREDIT:
4348 : : case ACCT_TYPE_LIABILITY:
4349 : : case ACCT_TYPE_INCOME:
4350 : : case ACCT_TYPE_EXPENSE:
4351 : : case ACCT_TYPE_EQUITY:
4352 : : return
4353 : : (1 << ACCT_TYPE_BANK) |
4354 : : (1 << ACCT_TYPE_CASH) |
4355 : : (1 << ACCT_TYPE_ASSET) |
4356 : : (1 << ACCT_TYPE_CREDIT) |
4357 : : (1 << ACCT_TYPE_LIABILITY) |
4358 : : (1 << ACCT_TYPE_INCOME) |
4359 : : (1 << ACCT_TYPE_EXPENSE) |
4360 : 8 : (1 << ACCT_TYPE_EQUITY);
4361 : 3 : case ACCT_TYPE_STOCK:
4362 : : case ACCT_TYPE_MUTUAL:
4363 : : case ACCT_TYPE_CURRENCY:
4364 : : return
4365 : : (1 << ACCT_TYPE_STOCK) |
4366 : : (1 << ACCT_TYPE_MUTUAL) |
4367 : 3 : (1 << ACCT_TYPE_CURRENCY);
4368 : 1 : case ACCT_TYPE_RECEIVABLE:
4369 : 1 : return (1 << ACCT_TYPE_RECEIVABLE);
4370 : 1 : case ACCT_TYPE_PAYABLE:
4371 : 1 : return (1 << ACCT_TYPE_PAYABLE);
4372 : 1 : case ACCT_TYPE_TRADING:
4373 : 1 : return (1 << ACCT_TYPE_TRADING);
4374 : 0 : default:
4375 : 0 : PERR("bad account type: %d", type);
4376 : 0 : return 0;
4377 : : }
4378 : : }
4379 : : guint32
4380 : 282 : xaccParentAccountTypesCompatibleWith (GNCAccountType type)
4381 : : {
4382 : 282 : switch (type)
4383 : : {
4384 : 200 : case ACCT_TYPE_BANK:
4385 : : case ACCT_TYPE_CASH:
4386 : : case ACCT_TYPE_ASSET:
4387 : : case ACCT_TYPE_STOCK:
4388 : : case ACCT_TYPE_MUTUAL:
4389 : : case ACCT_TYPE_CURRENCY:
4390 : : case ACCT_TYPE_CREDIT:
4391 : : case ACCT_TYPE_LIABILITY:
4392 : : case ACCT_TYPE_RECEIVABLE:
4393 : : case ACCT_TYPE_PAYABLE:
4394 : : return
4395 : : (1 << ACCT_TYPE_BANK) |
4396 : : (1 << ACCT_TYPE_CASH) |
4397 : : (1 << ACCT_TYPE_ASSET) |
4398 : : (1 << ACCT_TYPE_STOCK) |
4399 : : (1 << ACCT_TYPE_MUTUAL) |
4400 : : (1 << ACCT_TYPE_CURRENCY) |
4401 : : (1 << ACCT_TYPE_CREDIT) |
4402 : : (1 << ACCT_TYPE_LIABILITY) |
4403 : : (1 << ACCT_TYPE_RECEIVABLE) |
4404 : : (1 << ACCT_TYPE_PAYABLE) |
4405 : 200 : (1 << ACCT_TYPE_ROOT);
4406 : 40 : case ACCT_TYPE_INCOME:
4407 : : case ACCT_TYPE_EXPENSE:
4408 : : return
4409 : : (1 << ACCT_TYPE_INCOME) |
4410 : : (1 << ACCT_TYPE_EXPENSE) |
4411 : 40 : (1 << ACCT_TYPE_ROOT);
4412 : 20 : case ACCT_TYPE_EQUITY:
4413 : : return
4414 : : (1 << ACCT_TYPE_EQUITY) |
4415 : 20 : (1 << ACCT_TYPE_ROOT);
4416 : 20 : case ACCT_TYPE_TRADING:
4417 : : return
4418 : : (1 << ACCT_TYPE_TRADING) |
4419 : 20 : (1 << ACCT_TYPE_ROOT);
4420 : 2 : default:
4421 : 2 : PERR("bad account type: %d", type);
4422 : 2 : return 0;
4423 : : }
4424 : : }
4425 : :
4426 : : gboolean
4427 : 280 : xaccAccountTypesCompatible (GNCAccountType parent_type,
4428 : : GNCAccountType child_type)
4429 : : {
4430 : : /* ACCT_TYPE_NONE isn't compatible with anything, even ACCT_TYPE_NONE. */
4431 : 280 : if (parent_type == ACCT_TYPE_NONE || child_type == ACCT_TYPE_NONE)
4432 : 14 : return FALSE;
4433 : :
4434 : : /* ACCT_TYPE_ROOT can't have a parent account, and asking will raise
4435 : : * an error. */
4436 : 266 : if (child_type == ACCT_TYPE_ROOT)
4437 : 0 : return FALSE;
4438 : :
4439 : 266 : return ((xaccParentAccountTypesCompatibleWith (child_type) &
4440 : 266 : (1 << parent_type))
4441 : 266 : != 0);
4442 : : }
4443 : :
4444 : : guint32
4445 : 0 : xaccAccountTypesValid(void)
4446 : : {
4447 : 0 : guint32 mask = (1 << NUM_ACCOUNT_TYPES) - 1;
4448 : 0 : mask &= ~((1 << ACCT_TYPE_CURRENCY) | /* DEPRECATED */
4449 : : (1 << ACCT_TYPE_ROOT)); /* ROOT */
4450 : :
4451 : 0 : return mask;
4452 : : }
4453 : :
4454 : 0 : gboolean xaccAccountIsAssetLiabType(GNCAccountType t)
4455 : : {
4456 : 0 : switch (t)
4457 : : {
4458 : 0 : case ACCT_TYPE_RECEIVABLE:
4459 : : case ACCT_TYPE_PAYABLE:
4460 : 0 : return FALSE;
4461 : 0 : default:
4462 : 0 : return (xaccAccountTypesCompatible(ACCT_TYPE_ASSET, t)
4463 : 0 : || xaccAccountTypesCompatible(ACCT_TYPE_LIABILITY, t));
4464 : : }
4465 : : }
4466 : :
4467 : : GNCAccountType
4468 : 0 : xaccAccountTypeGetFundamental (GNCAccountType t)
4469 : : {
4470 : 0 : switch (t)
4471 : : {
4472 : 0 : case ACCT_TYPE_BANK:
4473 : : case ACCT_TYPE_STOCK:
4474 : : case ACCT_TYPE_MONEYMRKT:
4475 : : case ACCT_TYPE_CHECKING:
4476 : : case ACCT_TYPE_SAVINGS:
4477 : : case ACCT_TYPE_MUTUAL:
4478 : : case ACCT_TYPE_CURRENCY:
4479 : : case ACCT_TYPE_CASH:
4480 : : case ACCT_TYPE_ASSET:
4481 : : case ACCT_TYPE_RECEIVABLE:
4482 : 0 : return ACCT_TYPE_ASSET;
4483 : 0 : case ACCT_TYPE_CREDIT:
4484 : : case ACCT_TYPE_LIABILITY:
4485 : : case ACCT_TYPE_PAYABLE:
4486 : : case ACCT_TYPE_CREDITLINE:
4487 : 0 : return ACCT_TYPE_LIABILITY;
4488 : 0 : case ACCT_TYPE_INCOME:
4489 : 0 : return ACCT_TYPE_INCOME;
4490 : 0 : case ACCT_TYPE_EXPENSE:
4491 : 0 : return ACCT_TYPE_EXPENSE;
4492 : 0 : case ACCT_TYPE_EQUITY:
4493 : 0 : return ACCT_TYPE_EQUITY;
4494 : 0 : case ACCT_TYPE_TRADING:
4495 : : default:
4496 : 0 : return ACCT_TYPE_NONE;
4497 : : }
4498 : : }
4499 : :
4500 : 308 : gboolean xaccAccountIsAPARType(GNCAccountType t)
4501 : : {
4502 : 308 : switch (t)
4503 : : {
4504 : 123 : case ACCT_TYPE_RECEIVABLE:
4505 : : case ACCT_TYPE_PAYABLE:
4506 : 123 : return TRUE;
4507 : 185 : default:
4508 : 185 : return FALSE;
4509 : : }
4510 : : }
4511 : :
4512 : 0 : gboolean xaccAccountIsEquityType(GNCAccountType t)
4513 : : {
4514 : 0 : switch (t)
4515 : : {
4516 : 0 : case ACCT_TYPE_EQUITY:
4517 : 0 : return TRUE;
4518 : 0 : default:
4519 : 0 : return FALSE;
4520 : : }
4521 : : }
4522 : :
4523 : : gboolean
4524 : 549 : xaccAccountIsPriced(const Account *acc)
4525 : : {
4526 : : AccountPrivate *priv;
4527 : :
4528 : 549 : g_return_val_if_fail(GNC_IS_ACCOUNT(acc), FALSE);
4529 : :
4530 : 549 : priv = GET_PRIVATE(acc);
4531 : 1035 : return (priv->type == ACCT_TYPE_STOCK || priv->type == ACCT_TYPE_MUTUAL ||
4532 : 1035 : priv->type == ACCT_TYPE_CURRENCY);
4533 : : }
4534 : :
4535 : : /********************************************************************\
4536 : : \********************************************************************/
4537 : :
4538 : : gboolean
4539 : 1 : xaccAccountGetReconcileLastDate (const Account *acc, time64 *last_date)
4540 : : {
4541 : 1 : gboolean retval = FALSE;
4542 : 4 : auto date = get_kvp_int64_path (acc, {KEY_RECONCILE_INFO, "last-date"});
4543 : :
4544 : 1 : if (date)
4545 : : {
4546 : 1 : if (last_date)
4547 : 1 : *last_date = *date;
4548 : 1 : retval = TRUE;
4549 : : }
4550 : 1 : return retval;
4551 : 3 : }
4552 : :
4553 : : /********************************************************************\
4554 : : \********************************************************************/
4555 : :
4556 : : void
4557 : 1 : xaccAccountSetReconcileLastDate (Account *acc, time64 last_date)
4558 : : {
4559 : 5 : set_kvp_int64_path (acc, {KEY_RECONCILE_INFO, "last-date"}, last_date);
4560 : 3 : }
4561 : :
4562 : : /********************************************************************\
4563 : : \********************************************************************/
4564 : :
4565 : : gboolean
4566 : 2 : xaccAccountGetReconcileLastInterval (const Account *acc,
4567 : : int *months, int *days)
4568 : : {
4569 : 2 : if (!acc) return FALSE;
4570 : 10 : auto m{get_kvp_int64_path (acc, {KEY_RECONCILE_INFO, "last-interval", "months"})};
4571 : 10 : auto d{get_kvp_int64_path (acc, {KEY_RECONCILE_INFO, "last-interval", "days"})};
4572 : 2 : if (m && d)
4573 : : {
4574 : 1 : if (months)
4575 : 1 : *months = *m;
4576 : 1 : if (days)
4577 : 1 : *days = *d;
4578 : 1 : return true;
4579 : : }
4580 : 1 : return false;
4581 : 10 : }
4582 : :
4583 : : /********************************************************************\
4584 : : \********************************************************************/
4585 : :
4586 : : void
4587 : 1 : xaccAccountSetReconcileLastInterval (Account *acc, int months, int days)
4588 : : {
4589 : 6 : set_kvp_int64_path (acc, {KEY_RECONCILE_INFO, "last-interval", "months"}, months);
4590 : 5 : set_kvp_int64_path (acc, {KEY_RECONCILE_INFO, "last-interval", "days"}, days);
4591 : 5 : }
4592 : :
4593 : : /********************************************************************\
4594 : : \********************************************************************/
4595 : :
4596 : : gboolean
4597 : 1 : xaccAccountGetReconcilePostponeDate (const Account *acc, time64 *postpone_date)
4598 : : {
4599 : 6 : if (auto date = get_kvp_int64_path (acc, {KEY_RECONCILE_INFO, KEY_POSTPONE, "date"}))
4600 : : {
4601 : 1 : if (postpone_date)
4602 : 1 : *postpone_date = *date;
4603 : 1 : return true;
4604 : : }
4605 : 0 : return false;
4606 : 3 : }
4607 : :
4608 : : /********************************************************************\
4609 : : \********************************************************************/
4610 : :
4611 : : void
4612 : 1 : xaccAccountSetReconcilePostponeDate (Account *acc, time64 postpone_date)
4613 : : {
4614 : 6 : set_kvp_int64_path (acc, {KEY_RECONCILE_INFO, KEY_POSTPONE, "date"}, postpone_date);
4615 : 3 : }
4616 : :
4617 : : /********************************************************************\
4618 : : \********************************************************************/
4619 : :
4620 : : gboolean
4621 : 3 : xaccAccountGetReconcilePostponeBalance (const Account *acc,
4622 : : gnc_numeric *balance)
4623 : : {
4624 : 18 : if (auto bal = get_kvp_gnc_numeric_path (acc, {KEY_RECONCILE_INFO, KEY_POSTPONE, "balance"}))
4625 : : {
4626 : 1 : if (balance)
4627 : 1 : *balance = *bal;
4628 : 1 : return true;
4629 : : }
4630 : 2 : return false;
4631 : 9 : }
4632 : :
4633 : : /********************************************************************\
4634 : : \********************************************************************/
4635 : :
4636 : : void
4637 : 1 : xaccAccountSetReconcilePostponeBalance (Account *acc, gnc_numeric balance)
4638 : : {
4639 : 6 : set_kvp_gnc_numeric_path (acc, {KEY_RECONCILE_INFO, KEY_POSTPONE, "balance"}, balance);
4640 : 3 : }
4641 : :
4642 : : /********************************************************************\
4643 : :
4644 : : \********************************************************************/
4645 : :
4646 : : void
4647 : 1 : xaccAccountClearReconcilePostpone (Account *acc)
4648 : : {
4649 : 5 : set_kvp_gnc_numeric_path (acc, {KEY_RECONCILE_INFO, KEY_POSTPONE}, {});
4650 : 2 : }
4651 : :
4652 : : /********************************************************************\
4653 : : \********************************************************************/
4654 : :
4655 : : const char *
4656 : 6 : xaccAccountGetLastNum (const Account *acc)
4657 : : {
4658 : 18 : return get_kvp_string_path (acc, {"last-num"});
4659 : : }
4660 : :
4661 : : /********************************************************************\
4662 : : \********************************************************************/
4663 : :
4664 : : void
4665 : 385 : xaccAccountSetLastNum (Account *acc, const char *num)
4666 : : {
4667 : 770 : set_kvp_string_path (acc, {"last-num"}, num);
4668 : 385 : }
4669 : :
4670 : :
4671 : : /********************************************************************\
4672 : : \********************************************************************/
4673 : :
4674 : : static bool
4675 : 8 : get_balance_limit (const Account* acc, const std::string& key, gnc_numeric* balance)
4676 : : {
4677 : 32 : auto limit = get_kvp_gnc_numeric_path (acc, {KEY_BALANCE_LIMIT, key});
4678 : 8 : if (limit)
4679 : 4 : *balance = gnc_numeric_create (limit->num, limit->denom);
4680 : 16 : return limit.has_value();
4681 : 8 : }
4682 : :
4683 : : static void
4684 : 6 : set_balance_limit (Account *acc, const std::string& key, std::optional<gnc_numeric> balance)
4685 : : {
4686 : 6 : if (balance && gnc_numeric_check (*balance))
4687 : 2 : return;
4688 : 16 : set_kvp_gnc_numeric_path (acc, {KEY_BALANCE_LIMIT, key}, balance);
4689 : 4 : }
4690 : :
4691 : : gboolean
4692 : 4 : xaccAccountGetHigherBalanceLimit (const Account *acc,
4693 : : gnc_numeric *balance)
4694 : : {
4695 : 4 : return get_balance_limit (acc, KEY_BALANCE_HIGHER_LIMIT_VALUE, balance);
4696 : : }
4697 : :
4698 : : gboolean
4699 : 4 : xaccAccountGetLowerBalanceLimit (const Account *acc,
4700 : : gnc_numeric *balance)
4701 : : {
4702 : 4 : return get_balance_limit (acc, KEY_BALANCE_LOWER_LIMIT_VALUE, balance);
4703 : : }
4704 : :
4705 : : void
4706 : 2 : xaccAccountSetHigherBalanceLimit (Account *acc, gnc_numeric balance)
4707 : : {
4708 : 2 : set_balance_limit (acc, KEY_BALANCE_HIGHER_LIMIT_VALUE, balance);
4709 : 2 : }
4710 : :
4711 : : void
4712 : 2 : xaccAccountSetLowerBalanceLimit (Account *acc, gnc_numeric balance)
4713 : : {
4714 : 2 : set_balance_limit (acc, KEY_BALANCE_LOWER_LIMIT_VALUE, balance);
4715 : 2 : }
4716 : :
4717 : : void
4718 : 1 : xaccAccountClearHigherBalanceLimit (Account *acc)
4719 : : {
4720 : 1 : set_balance_limit (acc, KEY_BALANCE_HIGHER_LIMIT_VALUE, {});
4721 : 1 : }
4722 : :
4723 : : void
4724 : 1 : xaccAccountClearLowerBalanceLimit (Account *acc)
4725 : : {
4726 : 1 : set_balance_limit (acc, KEY_BALANCE_LOWER_LIMIT_VALUE, {});
4727 : 1 : }
4728 : :
4729 : : gboolean
4730 : 3 : xaccAccountGetIncludeSubAccountBalances (const Account *acc)
4731 : : {
4732 : 12 : return get_kvp_boolean_path (acc, {KEY_BALANCE_LIMIT, KEY_BALANCE_INCLUDE_SUB_ACCTS});
4733 : 3 : }
4734 : :
4735 : : void
4736 : 2 : xaccAccountSetIncludeSubAccountBalances (Account *acc, gboolean inc_sub)
4737 : : {
4738 : 8 : set_kvp_boolean_path (acc, {KEY_BALANCE_LIMIT, KEY_BALANCE_INCLUDE_SUB_ACCTS}, inc_sub);
4739 : 4 : }
4740 : :
4741 : : /********************************************************************\
4742 : : \********************************************************************/
4743 : :
4744 : : static Account *
4745 : 5 : GetOrMakeOrphanAccount (Account *root, gnc_commodity * currency)
4746 : : {
4747 : : char * accname;
4748 : : Account * acc;
4749 : :
4750 : 5 : g_return_val_if_fail (root, nullptr);
4751 : :
4752 : : /* build the account name */
4753 : 5 : if (!currency)
4754 : : {
4755 : 0 : PERR ("No currency specified!");
4756 : 0 : return nullptr;
4757 : : }
4758 : :
4759 : 5 : accname = g_strconcat (_("Orphaned Gains"), "-",
4760 : : gnc_commodity_get_mnemonic (currency), nullptr);
4761 : :
4762 : : /* See if we've got one of these going already ... */
4763 : 5 : acc = gnc_account_lookup_by_name(root, accname);
4764 : :
4765 : 5 : if (acc == nullptr)
4766 : : {
4767 : : /* Guess not. We'll have to build one. */
4768 : 3 : acc = xaccMallocAccount (gnc_account_get_book(root));
4769 : 3 : xaccAccountBeginEdit (acc);
4770 : 3 : xaccAccountSetName (acc, accname);
4771 : 3 : xaccAccountSetCommodity (acc, currency);
4772 : 3 : xaccAccountSetType (acc, ACCT_TYPE_INCOME);
4773 : 3 : xaccAccountSetDescription (acc, _("Realized Gain/Loss"));
4774 : 3 : xaccAccountSetNotes (acc,
4775 : 3 : _("Realized Gains or Losses from "
4776 : : "Commodity or Trading Accounts "
4777 : : "that haven't been recorded elsewhere."));
4778 : :
4779 : : /* Hang the account off the root. */
4780 : 3 : gnc_account_append_child (root, acc);
4781 : 3 : xaccAccountCommitEdit (acc);
4782 : : }
4783 : :
4784 : 5 : g_free (accname);
4785 : :
4786 : 5 : return acc;
4787 : : }
4788 : :
4789 : : Account *
4790 : 6 : xaccAccountGainsAccount (Account *acc, gnc_commodity *curr)
4791 : : {
4792 : 30 : Path path {KEY_LOT_MGMT, "gains-acct", gnc_commodity_get_unique_name (curr)};
4793 : 12 : auto gains_account = get_kvp_account_path (acc, path);
4794 : :
4795 : 6 : if (gains_account == nullptr) /* No gains account for this currency */
4796 : : {
4797 : 5 : gains_account = GetOrMakeOrphanAccount (gnc_account_get_root (acc), curr);
4798 : 5 : set_kvp_account_path (acc, path, gains_account);
4799 : : }
4800 : :
4801 : 6 : return gains_account;
4802 : 24 : }
4803 : :
4804 : : /********************************************************************\
4805 : : \********************************************************************/
4806 : :
4807 : : void
4808 : 16 : dxaccAccountSetPriceSrc(Account *acc, const char *src)
4809 : : {
4810 : 16 : if (!acc) return;
4811 : :
4812 : 16 : if (xaccAccountIsPriced(acc))
4813 : 15 : set_kvp_string_path (acc, {"old-price-source"}, src);
4814 : : }
4815 : :
4816 : : /********************************************************************\
4817 : : \********************************************************************/
4818 : :
4819 : : const char*
4820 : 428 : dxaccAccountGetPriceSrc(const Account *acc)
4821 : : {
4822 : 428 : if (!acc) return nullptr;
4823 : :
4824 : 428 : if (!xaccAccountIsPriced(acc)) return nullptr;
4825 : :
4826 : 57 : return get_kvp_string_path (acc, {"old-price-source"});
4827 : : }
4828 : :
4829 : : /********************************************************************\
4830 : : \********************************************************************/
4831 : :
4832 : : void
4833 : 16 : dxaccAccountSetQuoteTZ(Account *acc, const char *tz)
4834 : : {
4835 : 16 : if (!acc) return;
4836 : 16 : if (!xaccAccountIsPriced(acc)) return;
4837 : 15 : set_kvp_string_path (acc, {"old-quote-tz"}, tz);
4838 : : }
4839 : :
4840 : : /********************************************************************\
4841 : : \********************************************************************/
4842 : :
4843 : : const char*
4844 : 4 : dxaccAccountGetQuoteTZ(const Account *acc)
4845 : : {
4846 : 4 : if (!acc) return nullptr;
4847 : 4 : if (!xaccAccountIsPriced(acc)) return nullptr;
4848 : 12 : return get_kvp_string_path (acc, {"old-quote-tz"});
4849 : : }
4850 : :
4851 : : /********************************************************************\
4852 : : \********************************************************************/
4853 : :
4854 : : void
4855 : 1 : xaccAccountSetReconcileChildrenStatus(Account *acc, gboolean status)
4856 : : {
4857 : : /* Would have been nice to use G_TYPE_BOOLEAN, but the other
4858 : : * boolean kvps save the value as "true" or "false" and that would
4859 : : * be file-incompatible with this.
4860 : : */
4861 : 5 : set_kvp_int64_path (acc, {KEY_RECONCILE_INFO, KEY_INCLUDE_CHILDREN}, status);
4862 : 2 : }
4863 : :
4864 : : /********************************************************************\
4865 : : \********************************************************************/
4866 : :
4867 : : gboolean
4868 : 2 : xaccAccountGetReconcileChildrenStatus(const Account *acc)
4869 : : {
4870 : : /* access the account's kvp-data for status and return that, if no value
4871 : : * is found then we can assume not to include the children, that being
4872 : : * the default behaviour
4873 : : */
4874 : 8 : return get_kvp_boolean_path (acc, {KEY_RECONCILE_INFO, KEY_INCLUDE_CHILDREN});
4875 : 2 : }
4876 : :
4877 : : /********************************************************************\
4878 : : \********************************************************************/
4879 : :
4880 : : Split *
4881 : 2 : xaccAccountFindSplitByDesc(const Account *acc, const char *description)
4882 : : {
4883 : 10 : auto has_description = [description](const Split* s) -> bool
4884 : 10 : { return !g_strcmp0 (description, xaccTransGetDescription (xaccSplitGetParent (s))); };
4885 : 2 : return gnc_account_find_split (acc, has_description, true);
4886 : : }
4887 : :
4888 : : /* This routine is for finding a matching transaction in an account by
4889 : : * matching on the description field. [CAS: The rest of this comment
4890 : : * seems to belong somewhere else.] This routine is used for
4891 : : * auto-filling in registers with a default leading account. The
4892 : : * dest_trans is a transaction used for currency checking. */
4893 : : Transaction *
4894 : 1 : xaccAccountFindTransByDesc(const Account *acc, const char *description)
4895 : : {
4896 : 1 : auto split = xaccAccountFindSplitByDesc (acc, description);
4897 : 1 : return split ? xaccSplitGetParent (split) : nullptr;
4898 : : }
4899 : :
4900 : : /* ================================================================ */
4901 : : /* Concatenation, Merging functions */
4902 : :
4903 : : void
4904 : 1 : gnc_account_join_children (Account *to_parent, Account *from_parent)
4905 : : {
4906 : :
4907 : : /* errors */
4908 : 1 : g_return_if_fail(GNC_IS_ACCOUNT(to_parent));
4909 : 1 : g_return_if_fail(GNC_IS_ACCOUNT(from_parent));
4910 : :
4911 : : /* optimizations */
4912 : 1 : auto from_priv = GET_PRIVATE(from_parent);
4913 : 1 : if (from_priv->children.empty())
4914 : 0 : return;
4915 : :
4916 : 1 : ENTER (" ");
4917 : 1 : auto children = from_priv->children;
4918 : 3 : for (auto child : children)
4919 : 2 : gnc_account_append_child(to_parent, child);
4920 : 1 : LEAVE (" ");
4921 : 1 : }
4922 : : /********************************************************************\
4923 : : \********************************************************************/
4924 : :
4925 : : void
4926 : 3 : gnc_account_merge_children (Account *parent)
4927 : : {
4928 : 3 : g_return_if_fail(GNC_IS_ACCOUNT(parent));
4929 : :
4930 : 3 : auto ppriv = GET_PRIVATE(parent);
4931 : 12 : for (auto it_a = ppriv->children.begin(); it_a != ppriv->children.end(); it_a++)
4932 : : {
4933 : 9 : auto acc_a = *it_a;
4934 : 9 : auto priv_a = GET_PRIVATE(acc_a);
4935 : 23 : for (auto it_b = std::next(it_a); it_b != ppriv->children.end(); it_b++)
4936 : : {
4937 : 14 : auto acc_b = *it_b;
4938 : 14 : auto priv_b = GET_PRIVATE(acc_b);
4939 : 14 : if (0 != null_strcmp(priv_a->accountName, priv_b->accountName))
4940 : 13 : continue;
4941 : 1 : if (0 != null_strcmp(priv_a->accountCode, priv_b->accountCode))
4942 : 0 : continue;
4943 : 1 : if (0 != null_strcmp(priv_a->description, priv_b->description))
4944 : 0 : continue;
4945 : 1 : if (0 != null_strcmp(xaccAccountGetColor(acc_a),
4946 : : xaccAccountGetColor(acc_b)))
4947 : 0 : continue;
4948 : 1 : if (!gnc_commodity_equiv(priv_a->commodity, priv_b->commodity))
4949 : 0 : continue;
4950 : 1 : if (0 != null_strcmp(xaccAccountGetNotes(acc_a),
4951 : : xaccAccountGetNotes(acc_b)))
4952 : 0 : continue;
4953 : 1 : if (priv_a->type != priv_b->type)
4954 : 0 : continue;
4955 : :
4956 : : /* consolidate children */
4957 : 1 : if (!priv_b->children.empty())
4958 : : {
4959 : 1 : auto work = priv_b->children;
4960 : 3 : for (auto w : work)
4961 : 2 : gnc_account_append_child (acc_a, w);
4962 : :
4963 : 1 : qof_event_gen (&acc_a->inst, QOF_EVENT_MODIFY, nullptr);
4964 : 1 : qof_event_gen (&acc_b->inst, QOF_EVENT_MODIFY, nullptr);
4965 : 1 : }
4966 : :
4967 : : /* recurse to do the children's children */
4968 : 1 : gnc_account_merge_children (acc_a);
4969 : :
4970 : : /* consolidate transactions */
4971 : 1 : while (!priv_b->splits.empty())
4972 : 0 : xaccSplitSetAccount (priv_b->splits.front(), acc_a);
4973 : :
4974 : : /* move back one before removal. next iteration around the loop
4975 : : * will get the node after node_b */
4976 : 1 : it_b--;
4977 : :
4978 : : /* The destroy function will remove from list -- node_a is ok,
4979 : : * it's before node_b */
4980 : 1 : xaccAccountBeginEdit (acc_b);
4981 : 1 : xaccAccountDestroy (acc_b);
4982 : : }
4983 : : }
4984 : : }
4985 : :
4986 : : /* ================================================================ */
4987 : : /* Transaction Traversal functions */
4988 : :
4989 : :
4990 : : static void
4991 : 23 : xaccSplitsBeginStagedTransactionTraversals (SplitsVec& splits)
4992 : : {
4993 : 57 : for (auto s : splits)
4994 : : {
4995 : 34 : Transaction *trans = s->parent;
4996 : :
4997 : 34 : if (trans)
4998 : 34 : trans->marker = 0;
4999 : : }
5000 : 23 : }
5001 : :
5002 : : /* original function */
5003 : : void
5004 : 23 : xaccAccountBeginStagedTransactionTraversals (const Account *account)
5005 : : {
5006 : 23 : if (!account)
5007 : 0 : return;
5008 : 23 : xaccSplitsBeginStagedTransactionTraversals(GET_PRIVATE (account)->splits);
5009 : : }
5010 : :
5011 : : gboolean
5012 : 0 : xaccTransactionTraverse (Transaction *trans, int stage)
5013 : : {
5014 : 0 : if (trans == nullptr) return FALSE;
5015 : :
5016 : 0 : if (trans->marker < stage)
5017 : : {
5018 : 0 : trans->marker = stage;
5019 : 0 : return TRUE;
5020 : : }
5021 : :
5022 : 0 : return FALSE;
5023 : : }
5024 : :
5025 : : /* Replacement for xaccGroupBeginStagedTransactionTraversals */
5026 : : void
5027 : 55 : gnc_account_tree_begin_staged_transaction_traversals (Account *account)
5028 : : {
5029 : 1260 : auto do_one_account = [](auto acc)
5030 : 1260 : { gnc_account_foreach_split (acc, [](auto s){ s->parent->marker = 0; }); };
5031 : 55 : gnc_account_foreach_descendant (account, do_one_account);
5032 : 55 : }
5033 : :
5034 : : int
5035 : 23 : xaccAccountStagedTransactionTraversal (const Account *acc,
5036 : : unsigned int stage,
5037 : : TransactionCallback thunk,
5038 : : void *cb_data)
5039 : : {
5040 : 23 : if (!acc) return 0;
5041 : :
5042 : : // iterate on copy of splits. some callers modify the splitsvec.
5043 : 23 : auto splits = GET_PRIVATE(acc)->splits;
5044 : 50 : for (auto s : splits)
5045 : : {
5046 : 28 : auto trans = s->parent;
5047 : 28 : if (trans && (trans->marker < stage))
5048 : : {
5049 : 20 : trans->marker = stage;
5050 : 20 : if (thunk)
5051 : : {
5052 : 20 : auto retval = thunk(trans, cb_data);
5053 : 20 : if (retval) return retval;
5054 : : }
5055 : : }
5056 : : }
5057 : :
5058 : 22 : return 0;
5059 : 23 : }
5060 : :
5061 : : int
5062 : 1307 : gnc_account_tree_staged_transaction_traversal (const Account *acc,
5063 : : unsigned int stage,
5064 : : TransactionCallback thunk,
5065 : : void *cb_data)
5066 : : {
5067 : : const AccountPrivate *priv;
5068 : : Transaction *trans;
5069 : : int retval;
5070 : :
5071 : 1307 : if (!acc) return 0;
5072 : :
5073 : : /* depth first traversal */
5074 : 1307 : priv = GET_PRIVATE(acc);
5075 : 2556 : for (auto acc_p : priv->children)
5076 : : {
5077 : 1252 : retval = gnc_account_tree_staged_transaction_traversal(acc_p, stage, thunk, cb_data);
5078 : 1252 : if (retval) return retval;
5079 : : }
5080 : :
5081 : : /* Now this account */
5082 : 3393 : for (auto s : priv->splits)
5083 : : {
5084 : 2090 : trans = s->parent;
5085 : 2090 : if (trans && (trans->marker < stage))
5086 : : {
5087 : 1013 : trans->marker = stage;
5088 : 1013 : if (thunk)
5089 : : {
5090 : 1013 : retval = thunk(trans, cb_data);
5091 : 1013 : if (retval) return retval;
5092 : : }
5093 : : }
5094 : : }
5095 : :
5096 : 1303 : return 0;
5097 : : }
5098 : :
5099 : : time64
5100 : 0 : gnc_account_get_earliest_date (const Account* account)
5101 : : {
5102 : 0 : g_return_val_if_fail (GNC_IS_ACCOUNT(account), INT64_MAX);
5103 : 0 : const auto& splits = xaccAccountGetSplits (account);
5104 : 0 : return splits.empty() ? INT64_MAX : xaccTransGetDate (xaccSplitGetParent (splits.front()));
5105 : : }
5106 : :
5107 : : /********************************************************************\
5108 : : \********************************************************************/
5109 : :
5110 : : int
5111 : 55 : xaccAccountTreeForEachTransaction (Account *acc,
5112 : : int (*proc)(Transaction *t, void *data),
5113 : : void *data)
5114 : : {
5115 : 55 : if (!acc || !proc) return 0;
5116 : :
5117 : 55 : gnc_account_tree_begin_staged_transaction_traversals (acc);
5118 : 55 : return gnc_account_tree_staged_transaction_traversal (acc, 42, proc, data);
5119 : : }
5120 : :
5121 : :
5122 : : gint
5123 : 23 : xaccAccountForEachTransaction(const Account *acc, TransactionCallback proc,
5124 : : void *data)
5125 : : {
5126 : 23 : if (!acc || !proc) return 0;
5127 : 23 : xaccAccountBeginStagedTransactionTraversals (acc);
5128 : 23 : return xaccAccountStagedTransactionTraversal(acc, 42, proc, data);
5129 : : }
5130 : :
5131 : : /* ================================================================ */
5132 : : /* The following functions are used by
5133 : : * src/import-export/import-backend.c to manipulate the contra-account
5134 : : * matching data. See src/import-export/import-backend.c for explanations.
5135 : : */
5136 : :
5137 : : #define IMAP_FRAME "import-map"
5138 : : #define IMAP_FRAME_BAYES "import-map-bayes"
5139 : :
5140 : : /* Look up an Account in the map */
5141 : : Account*
5142 : 7 : gnc_account_imap_find_account (Account *acc,
5143 : : const char *category,
5144 : : const char *key)
5145 : : {
5146 : 7 : if (!acc || !key) return nullptr;
5147 : 18 : std::vector<std::string> path {IMAP_FRAME};
5148 : 6 : if (category)
5149 : 8 : path.push_back (category);
5150 : 6 : path.push_back (key);
5151 : 6 : return get_kvp_account_path (acc, path);
5152 : 24 : }
5153 : :
5154 : : Account*
5155 : 0 : gnc_account_imap_find_any (QofBook *book, const char* category, const char *key)
5156 : : {
5157 : 0 : Account *account = nullptr;
5158 : :
5159 : : /* Get list of Accounts */
5160 : 0 : auto root = gnc_book_get_root_account (book);
5161 : 0 : auto accts = gnc_account_get_descendants_sorted (root);
5162 : :
5163 : : /* Go through list of accounts */
5164 : 0 : for (auto ptr = accts; ptr; ptr = g_list_next (ptr))
5165 : : {
5166 : 0 : auto tmp_acc = static_cast<Account*> (ptr->data);
5167 : :
5168 : 0 : if (gnc_account_imap_find_account (tmp_acc, category, key))
5169 : : {
5170 : 0 : account = tmp_acc;
5171 : 0 : break;
5172 : : }
5173 : : }
5174 : 0 : g_list_free (accts);
5175 : :
5176 : 0 : return account;
5177 : : }
5178 : :
5179 : : /* Store an Account in the map */
5180 : : void
5181 : 9 : gnc_account_imap_add_account (Account *acc,
5182 : : const char *category,
5183 : : const char *key,
5184 : : Account *added_acc)
5185 : : {
5186 : 9 : if (!acc || !key || !added_acc || !*key) return;
5187 : :
5188 : 42 : auto path = category ? Path{IMAP_FRAME, category, key} : Path{IMAP_FRAME, key};
5189 : :
5190 : 7 : set_kvp_account_path (acc, path, added_acc);
5191 : 35 : }
5192 : :
5193 : : /* Remove a reference to an Account in the map */
5194 : : void
5195 : 4 : gnc_account_imap_delete_account (Account *acc,
5196 : : const char *category,
5197 : : const char *key)
5198 : : {
5199 : 4 : if (!acc || !key) return;
5200 : :
5201 : 18 : auto path = category ? Path{IMAP_FRAME, category, key} : Path{IMAP_FRAME, key};
5202 : 3 : if (qof_instance_has_path_slot (QOF_INSTANCE (acc), path))
5203 : : {
5204 : 3 : qof_instance_slot_path_delete (QOF_INSTANCE (acc), path);
5205 : 3 : if (category)
5206 : 8 : qof_instance_slot_path_delete_if_empty (QOF_INSTANCE (acc), {IMAP_FRAME, category});
5207 : 9 : qof_instance_slot_path_delete_if_empty (QOF_INSTANCE (acc), {IMAP_FRAME});
5208 : : }
5209 : 3 : qof_instance_set_dirty (QOF_INSTANCE (acc));
5210 : 24 : }
5211 : :
5212 : : /*--------------------------------------------------------------------------
5213 : : Below here is the bayes transaction to account matching system
5214 : : --------------------------------------------------------------------------*/
5215 : :
5216 : :
5217 : : /** intermediate values used to calculate the bayes probability of a given account
5218 : : where p(AB) = (a*b)/[a*b + (1-a)(1-b)], product is (a*b),
5219 : : product_difference is (1-a) * (1-b)
5220 : : */
5221 : : struct AccountProbability
5222 : : {
5223 : : double product; /* product of probabilities */
5224 : : double product_difference; /* product of (1-probabilities) */
5225 : : };
5226 : :
5227 : : struct AccountTokenCount
5228 : : {
5229 : : std::string account_guid;
5230 : : int64_t token_count; /** occurrences of a given token for this account_guid */
5231 : : };
5232 : :
5233 : : /** total_count and the token_count for a given account let us calculate the
5234 : : * probability of a given account with any single token
5235 : : */
5236 : : struct TokenAccountsInfo
5237 : : {
5238 : : std::vector<AccountTokenCount> accounts;
5239 : : int64_t total_count;
5240 : : };
5241 : :
5242 : : /** holds an account guid and its corresponding integer probability
5243 : : the integer probability is some factor of 10
5244 : : */
5245 : : struct AccountInfo
5246 : : {
5247 : : std::string account_guid;
5248 : : int32_t probability;
5249 : : };
5250 : :
5251 : : static void
5252 : 11 : build_token_info(char const * suffix, KvpValue * value, TokenAccountsInfo & tokenInfo)
5253 : : {
5254 : 11 : if (strlen(suffix) == GUID_ENCODING_LENGTH)
5255 : : {
5256 : 10 : tokenInfo.total_count += value->get<int64_t>();
5257 : : /*By convention, the key ends with the account GUID.*/
5258 : 30 : tokenInfo.accounts.emplace_back(AccountTokenCount{std::string{suffix}, value->get<int64_t>()});
5259 : : }
5260 : 11 : }
5261 : :
5262 : : /** We scale the probability values by probability_factor.
5263 : : ie. with probability_factor of 100000, 10% would be
5264 : : 0.10 * 100000 = 10000 */
5265 : : static constexpr int probability_factor = 100000;
5266 : :
5267 : : static FinalProbabilityVec
5268 : 8 : build_probabilities(ProbabilityVec const & first_pass)
5269 : : {
5270 : 8 : FinalProbabilityVec ret;
5271 : 16 : for (auto const & first_pass_prob : first_pass)
5272 : : {
5273 : 8 : auto const & account_probability = first_pass_prob.second;
5274 : : /* P(AB) = A*B / [A*B + (1-A)*(1-B)]
5275 : : * NOTE: so we only keep track of a running product(A*B*C...)
5276 : : * and product difference ((1-A)(1-B)...)
5277 : : */
5278 : 8 : int32_t probability = (account_probability.product /
5279 : 8 : (account_probability.product + account_probability.product_difference)) * probability_factor;
5280 : 8 : ret.push_back({first_pass_prob.first, probability});
5281 : : }
5282 : 8 : return ret;
5283 : 0 : }
5284 : :
5285 : : static AccountInfo
5286 : 8 : highest_probability(FinalProbabilityVec const & probabilities)
5287 : : {
5288 : 16 : AccountInfo ret {"", std::numeric_limits<int32_t>::min()};
5289 : 16 : for (auto const & prob : probabilities)
5290 : 8 : if (prob.second > ret.probability)
5291 : 8 : ret = AccountInfo {prob.first, prob.second};
5292 : 8 : return ret;
5293 : 0 : }
5294 : :
5295 : : static ProbabilityVec
5296 : 9 : get_first_pass_probabilities(Account* acc, GList * tokens)
5297 : : {
5298 : 9 : ProbabilityVec ret;
5299 : : /* find the probability for each account that contains any of the tokens
5300 : : * in the input tokens list. */
5301 : 21 : for (auto current_token = tokens; current_token; current_token = current_token->next)
5302 : : {
5303 : 12 : TokenAccountsInfo tokenInfo{};
5304 : 24 : auto path = std::string{IMAP_FRAME_BAYES "/"} + static_cast <char const *> (current_token->data) + "/";
5305 : 12 : qof_instance_foreach_slot_prefix (QOF_INSTANCE (acc), path, &build_token_info, tokenInfo);
5306 : 22 : for (auto const & current_account_token : tokenInfo.accounts)
5307 : : {
5308 : 10 : auto item = std::find_if(ret.begin(), ret.end(), [¤t_account_token]
5309 : : (std::pair<std::string, AccountProbability> const & a) {
5310 : 2 : return current_account_token.account_guid == a.first;
5311 : : });
5312 : 10 : if (item != ret.end())
5313 : : {/* This account is already in the map */
5314 : 6 : item->second.product = ((double)current_account_token.token_count /
5315 : 2 : (double)tokenInfo.total_count) * item->second.product;
5316 : 2 : item->second.product_difference = ((double)1 - ((double)current_account_token.token_count /
5317 : 2 : (double)tokenInfo.total_count)) * item->second.product_difference;
5318 : : }
5319 : : else
5320 : : {
5321 : : /* add a new entry */
5322 : : AccountProbability new_probability;
5323 : 8 : new_probability.product = ((double)current_account_token.token_count /
5324 : 8 : (double)tokenInfo.total_count);
5325 : 8 : new_probability.product_difference = 1 - (new_probability.product);
5326 : 8 : ret.push_back({current_account_token.account_guid, std::move(new_probability)});
5327 : : }
5328 : : } /* for all accounts in tokenInfo */
5329 : 12 : }
5330 : 9 : return ret;
5331 : 0 : }
5332 : :
5333 : : static std::string
5334 : 5 : look_for_old_separator_descendants (Account *root, std::string const & full_name, const gchar *separator)
5335 : : {
5336 : : GList *top_accounts, *ptr;
5337 : 5 : gint found_len = 0;
5338 : : gchar found_sep;
5339 : 5 : top_accounts = gnc_account_get_descendants (root);
5340 : 5 : PINFO("Incoming full_name is '%s', current separator is '%s'", full_name.c_str (), separator);
5341 : : /* Go through list of top level accounts */
5342 : 40 : for (ptr = top_accounts; ptr; ptr = g_list_next (ptr))
5343 : : {
5344 : 35 : const gchar *name = xaccAccountGetName (static_cast <Account const *> (ptr->data));
5345 : : // we are looking for the longest top level account that matches
5346 : 35 : if (g_str_has_prefix (full_name.c_str (), name))
5347 : : {
5348 : 6 : gint name_len = strlen (name);
5349 : 6 : const gchar old_sep = full_name[name_len];
5350 : 6 : if (!g_ascii_isalnum (old_sep)) // test for non alpha numeric
5351 : : {
5352 : 6 : if (name_len > found_len)
5353 : : {
5354 : 6 : found_sep = full_name[name_len];
5355 : 6 : found_len = name_len;
5356 : : }
5357 : : }
5358 : : }
5359 : : }
5360 : 5 : g_list_free (top_accounts); // Free the List
5361 : 5 : std::string new_name {full_name};
5362 : 5 : if (found_len > 1)
5363 : 5 : std::replace (new_name.begin (), new_name.end (), found_sep, *separator);
5364 : 5 : PINFO ("Return full_name is '%s'", new_name.c_str ());
5365 : 10 : return new_name;
5366 : 0 : }
5367 : :
5368 : : static std::string
5369 : 6 : get_guid_from_account_name (Account * root, std::string const & name)
5370 : : {
5371 : 6 : auto map_account = gnc_account_lookup_by_full_name (root, name.c_str ());
5372 : 6 : if (!map_account)
5373 : : {
5374 : : auto temp_account_name = look_for_old_separator_descendants (root, name,
5375 : 5 : gnc_get_account_separator_string ());
5376 : 5 : map_account = gnc_account_lookup_by_full_name (root, temp_account_name.c_str ());
5377 : 5 : }
5378 : 6 : auto temp_guid = gnc::GUID {*xaccAccountGetGUID (map_account)};
5379 : 12 : return temp_guid.to_string ();
5380 : : }
5381 : :
5382 : : static FlatKvpEntry
5383 : 7 : convert_entry (KvpEntry entry, Account* root)
5384 : : {
5385 : : /*We need to make a copy here.*/
5386 : 7 : auto account_name = entry.first.back();
5387 : 7 : if (!gnc::GUID::is_valid_guid (account_name))
5388 : : {
5389 : : /* Earlier version stored the account name in the import map, and
5390 : : * there were early beta versions of 2.7 that stored a GUID.
5391 : : * If there is no GUID, we assume it's an account name. */
5392 : : /* Take off the account name and replace it with the GUID */
5393 : 6 : entry.first.pop_back();
5394 : 6 : auto guid_str = get_guid_from_account_name (root, account_name);
5395 : 6 : entry.first.emplace_back (guid_str);
5396 : 6 : }
5397 : 7 : std::string new_key {std::accumulate (entry.first.begin(), entry.first.end(), std::string {})};
5398 : 7 : new_key = IMAP_FRAME_BAYES + new_key;
5399 : 14 : return {new_key, entry.second};
5400 : 7 : }
5401 : :
5402 : : static std::vector<FlatKvpEntry>
5403 : 2 : get_flat_imap (Account * acc)
5404 : : {
5405 : 2 : auto frame = qof_instance_get_slots (QOF_INSTANCE (acc));
5406 : 4 : auto slot = frame->get_slot ({IMAP_FRAME_BAYES});
5407 : 2 : if (!slot)
5408 : 1 : return {};
5409 : 1 : auto imap_frame = slot->get<KvpFrame*> ();
5410 : 1 : auto flat_kvp = imap_frame->flatten_kvp ();
5411 : 1 : auto root = gnc_account_get_root (acc);
5412 : 1 : std::vector <FlatKvpEntry> ret;
5413 : 8 : for (auto const & flat_entry : flat_kvp)
5414 : : {
5415 : 7 : auto converted_entry = convert_entry (flat_entry, root);
5416 : : /*If the entry was invalid, we don't perpetuate it.*/
5417 : 7 : if (converted_entry.first.size())
5418 : 7 : ret.emplace_back (converted_entry);
5419 : 7 : }
5420 : 1 : return ret;
5421 : 1 : }
5422 : :
5423 : : static bool
5424 : 14 : convert_imap_account_bayes_to_flat (Account *acc)
5425 : : {
5426 : 14 : auto frame = qof_instance_get_slots (QOF_INSTANCE (acc));
5427 : 14 : if (!frame->get_keys().size())
5428 : 12 : return false;
5429 : 2 : auto flat_imap = get_flat_imap(acc);
5430 : 2 : if (!flat_imap.size ())
5431 : 1 : return false;
5432 : 1 : xaccAccountBeginEdit(acc);
5433 : 2 : frame->set({IMAP_FRAME_BAYES}, nullptr);
5434 : 1 : std::for_each(flat_imap.begin(), flat_imap.end(),
5435 : 7 : [&frame] (FlatKvpEntry const & entry) {
5436 : 21 : frame->set({entry.first.c_str()}, entry.second);
5437 : 21 : });
5438 : 1 : qof_instance_set_dirty (QOF_INSTANCE (acc));
5439 : 1 : xaccAccountCommitEdit(acc);
5440 : 1 : return true;
5441 : 2 : }
5442 : :
5443 : : /*
5444 : : * Checks for import map data and converts them when found.
5445 : : */
5446 : : static bool
5447 : 2 : imap_convert_bayes_to_flat (QofBook * book)
5448 : : {
5449 : 2 : auto root = gnc_book_get_root_account (book);
5450 : 2 : auto accts = gnc_account_get_descendants_sorted (root);
5451 : 2 : bool ret = false;
5452 : 16 : for (auto ptr = accts; ptr; ptr = g_list_next (ptr))
5453 : : {
5454 : 14 : Account *acc = static_cast <Account*> (ptr->data);
5455 : 14 : if (convert_imap_account_bayes_to_flat (acc))
5456 : : {
5457 : 1 : ret = true;
5458 : 1 : gnc_features_set_used (book, GNC_FEATURE_GUID_FLAT_BAYESIAN);
5459 : : }
5460 : : }
5461 : 2 : g_list_free (accts);
5462 : 2 : return ret;
5463 : : }
5464 : :
5465 : : void
5466 : 1 : gnc_account_reset_convert_bayes_to_flat (void)
5467 : : {
5468 : 1 : imap_convert_bayes_to_flat_run = false;
5469 : 1 : }
5470 : :
5471 : : /*
5472 : : * Here we check to see the state of import map data.
5473 : : *
5474 : : * If the GUID_FLAT_BAYESIAN feature flag is set, everything
5475 : : * should be fine.
5476 : : *
5477 : : * If it is not set, there are two possibilities: import data
5478 : : * are present from a previous version or not. If they are,
5479 : : * they are converted, and the feature flag set. If there are
5480 : : * no previous data, nothing is done.
5481 : : */
5482 : : static void
5483 : 21 : check_import_map_data (QofBook *book)
5484 : : {
5485 : 21 : if (gnc_features_check_used (book, GNC_FEATURE_GUID_FLAT_BAYESIAN) ||
5486 : : imap_convert_bayes_to_flat_run)
5487 : 19 : return;
5488 : :
5489 : : /* This function will set GNC_FEATURE_GUID_FLAT_BAYESIAN if necessary.*/
5490 : 2 : imap_convert_bayes_to_flat (book);
5491 : 2 : imap_convert_bayes_to_flat_run = true;
5492 : : }
5493 : :
5494 : : static constexpr double threshold = .90 * probability_factor; /* 90% */
5495 : :
5496 : : /** Look up an Account in the map */
5497 : : Account*
5498 : 9 : gnc_account_imap_find_account_bayes (Account *acc, GList *tokens)
5499 : : {
5500 : 9 : if (!acc)
5501 : 0 : return nullptr;
5502 : 9 : auto book = gnc_account_get_book(acc);
5503 : 9 : check_import_map_data (book);
5504 : 9 : auto first_pass = get_first_pass_probabilities(acc, tokens);
5505 : 9 : if (!first_pass.size())
5506 : 1 : return nullptr;
5507 : 8 : auto final_probabilities = build_probabilities(first_pass);
5508 : 8 : if (!final_probabilities.size())
5509 : 0 : return nullptr;
5510 : 8 : auto best = highest_probability(final_probabilities);
5511 : 8 : if (best.account_guid == "")
5512 : 0 : return nullptr;
5513 : 8 : if (best.probability < threshold)
5514 : 0 : return nullptr;
5515 : : gnc::GUID guid;
5516 : : try {
5517 : 8 : guid = gnc::GUID::from_string(best.account_guid);
5518 : 0 : } catch (gnc::guid_syntax_exception&) {
5519 : 0 : return nullptr;
5520 : 0 : }
5521 : 8 : auto account = xaccAccountLookup (reinterpret_cast<GncGUID*>(&guid), book);
5522 : 8 : return account;
5523 : 9 : }
5524 : :
5525 : : static void
5526 : 14 : change_imap_entry (Account *acc, std::string const & path, int64_t token_count)
5527 : : {
5528 : 14 : PINFO("Source Account is '%s', Count is '%" G_GINT64_FORMAT "'",
5529 : : xaccAccountGetName (acc), token_count);
5530 : :
5531 : : // check for existing guid entry
5532 : 42 : if (auto existing_token_count = get_kvp_int64_path (acc, {path}))
5533 : : {
5534 : 5 : PINFO("found existing value of '%" G_GINT64_FORMAT "'", *existing_token_count);
5535 : 5 : token_count += *existing_token_count;
5536 : : }
5537 : :
5538 : : // Add or Update the entry based on guid
5539 : 56 : set_kvp_int64_path (acc, {path}, token_count);
5540 : 42 : }
5541 : :
5542 : : /** Updates the imap for a given account using a list of tokens */
5543 : : void
5544 : 11 : gnc_account_imap_add_account_bayes (Account *acc,
5545 : : GList *tokens,
5546 : : Account *added_acc)
5547 : : {
5548 : : GList *current_token;
5549 : : gint64 token_count;
5550 : : char *account_fullname;
5551 : : char *guid_string;
5552 : :
5553 : 11 : ENTER(" ");
5554 : 11 : if (!acc)
5555 : : {
5556 : 0 : LEAVE(" ");
5557 : 0 : return;
5558 : : }
5559 : 11 : check_import_map_data (gnc_account_get_book(acc));
5560 : :
5561 : 11 : g_return_if_fail (added_acc != nullptr);
5562 : 10 : account_fullname = gnc_account_get_full_name(added_acc);
5563 : 10 : xaccAccountBeginEdit (acc);
5564 : :
5565 : 10 : PINFO("account name: '%s'", account_fullname);
5566 : :
5567 : 10 : guid_string = guid_to_string (xaccAccountGetGUID (added_acc));
5568 : :
5569 : : /* process each token in the list */
5570 : 24 : for (current_token = g_list_first(tokens); current_token;
5571 : 14 : current_token = current_token->next)
5572 : : {
5573 : 14 : char* token = static_cast<char*>(current_token->data);
5574 : : /* Jump to next iteration if the pointer is not valid or if the
5575 : : string is empty. In HBCI import we almost always get an empty
5576 : : string, which doesn't work in the kvp loopkup later. So we
5577 : : skip this case here. */
5578 : 14 : if (!token || !token[0])
5579 : 0 : continue;
5580 : : /* start off with one token for this account */
5581 : 14 : token_count = 1;
5582 : 14 : PINFO("adding token '%s'", token);
5583 : 14 : auto path = std::string {IMAP_FRAME_BAYES} + '/' + token + '/' + guid_string;
5584 : : /* change the imap entry for the account */
5585 : 14 : change_imap_entry (acc, path, token_count);
5586 : 14 : }
5587 : : /* free up the account fullname and guid string */
5588 : 10 : xaccAccountCommitEdit (acc);
5589 : 10 : gnc_features_set_used (gnc_account_get_book(acc), GNC_FEATURE_GUID_FLAT_BAYESIAN);
5590 : 10 : g_free (account_fullname);
5591 : 10 : g_free (guid_string);
5592 : 10 : LEAVE(" ");
5593 : : }
5594 : :
5595 : : /*******************************************************************************/
5596 : :
5597 : : static void
5598 : 0 : build_non_bayes (const char *key, const GValue *value, gpointer user_data)
5599 : : {
5600 : 0 : if (!G_VALUE_HOLDS_BOXED (value))
5601 : 0 : return;
5602 : : QofBook *book;
5603 : 0 : GncGUID *guid = nullptr;
5604 : 0 : gchar *guid_string = nullptr;
5605 : 0 : auto imapInfo = (GncImapInfo*)user_data;
5606 : : // Get the book
5607 : 0 : book = qof_instance_get_book (imapInfo->source_account);
5608 : :
5609 : 0 : guid = (GncGUID*)g_value_get_boxed (value);
5610 : 0 : guid_string = guid_to_string (guid);
5611 : :
5612 : 0 : PINFO("build_non_bayes: match string '%s', match account guid: '%s'",
5613 : : (char*)key, guid_string);
5614 : :
5615 : 0 : auto imapInfo_node = static_cast <GncImapInfo*> (g_malloc(sizeof(GncImapInfo)));
5616 : :
5617 : 0 : imapInfo_node->source_account = imapInfo->source_account;
5618 : 0 : imapInfo_node->map_account = xaccAccountLookup (guid, book);
5619 : 0 : imapInfo_node->head = g_strdup (imapInfo->head);
5620 : 0 : imapInfo_node->match_string = g_strdup (key);
5621 : 0 : imapInfo_node->category = g_strdup (imapInfo->category);
5622 : 0 : imapInfo_node->count = g_strdup (" ");
5623 : :
5624 : 0 : imapInfo->list = g_list_prepend (imapInfo->list, imapInfo_node);
5625 : :
5626 : 0 : g_free (guid_string);
5627 : : }
5628 : :
5629 : : static void
5630 : 1 : build_bayes (const char *suffix, KvpValue * value, GncImapInfo & imapInfo)
5631 : : {
5632 : 1 : size_t guid_start = strlen(suffix) - GUID_ENCODING_LENGTH;
5633 : 2 : std::string account_guid {&suffix[guid_start]};
5634 : : GncGUID guid;
5635 : : try
5636 : : {
5637 : 1 : guid = gnc::GUID::from_string (account_guid);
5638 : : }
5639 : 0 : catch (const gnc::guid_syntax_exception& err)
5640 : : {
5641 : 0 : PWARN("Invalid GUID string from %s%s", IMAP_FRAME_BAYES, suffix);
5642 : 0 : }
5643 : 1 : auto map_account = xaccAccountLookup (&guid, gnc_account_get_book (imapInfo.source_account));
5644 : 1 : auto imap_node = static_cast <GncImapInfo*> (g_malloc (sizeof (GncImapInfo)));
5645 : 1 : auto count = value->get <int64_t> ();
5646 : 1 : imap_node->source_account = imapInfo.source_account;
5647 : 1 : imap_node->map_account = map_account;
5648 : 1 : imap_node->head = g_strdup_printf ("%s%s", IMAP_FRAME_BAYES, suffix);
5649 : 1 : imap_node->match_string = g_strndup (&suffix[1], guid_start - 2);
5650 : 1 : imap_node->category = g_strdup(" ");
5651 : 1 : imap_node->count = g_strdup_printf ("%" G_GINT64_FORMAT, count);
5652 : 1 : imapInfo.list = g_list_prepend (imapInfo.list, imap_node);
5653 : 1 : }
5654 : :
5655 : 1 : void gnc_account_imap_info_destroy (GncImapInfo* imapInfo)
5656 : : {
5657 : 1 : g_free (imapInfo->head);
5658 : 1 : g_free (imapInfo->category);
5659 : 1 : g_free (imapInfo->match_string);
5660 : 1 : g_free (imapInfo->count);
5661 : 1 : g_free (imapInfo);
5662 : 1 : }
5663 : :
5664 : : GList *
5665 : 1 : gnc_account_imap_get_info_bayes (Account *acc)
5666 : : {
5667 : 1 : check_import_map_data (gnc_account_get_book (acc));
5668 : : /* A dummy object which is used to hold the specified account, and the list
5669 : : * of data about which we care. */
5670 : 1 : GncImapInfo imapInfo {acc, nullptr};
5671 : 2 : qof_instance_foreach_slot_prefix (QOF_INSTANCE (acc), IMAP_FRAME_BAYES, &build_bayes, imapInfo);
5672 : 2 : return g_list_reverse(imapInfo.list);
5673 : : }
5674 : :
5675 : : GList *
5676 : 0 : gnc_account_imap_get_info (Account *acc, const char *category)
5677 : : {
5678 : 0 : GList *list = nullptr;
5679 : :
5680 : : GncImapInfo imapInfo;
5681 : :
5682 : 0 : std::vector<std::string> path {IMAP_FRAME};
5683 : 0 : if (category)
5684 : 0 : path.emplace_back (category);
5685 : :
5686 : 0 : imapInfo.source_account = acc;
5687 : 0 : imapInfo.list = list;
5688 : :
5689 : 0 : imapInfo.head = g_strdup (IMAP_FRAME);
5690 : 0 : imapInfo.category = g_strdup (category);
5691 : :
5692 : 0 : if (qof_instance_has_path_slot (QOF_INSTANCE (acc), path))
5693 : : {
5694 : 0 : qof_instance_foreach_slot (QOF_INSTANCE(acc), IMAP_FRAME, category,
5695 : : build_non_bayes, &imapInfo);
5696 : : }
5697 : 0 : g_free (imapInfo.head);
5698 : 0 : g_free (imapInfo.category);
5699 : 0 : return g_list_reverse(imapInfo.list);
5700 : 0 : }
5701 : :
5702 : : /*******************************************************************************/
5703 : :
5704 : : gchar *
5705 : 9 : gnc_account_get_map_entry (Account *acc, const char *head, const char *category)
5706 : : {
5707 : 69 : return g_strdup (category ?
5708 : : get_kvp_string_path (acc, {head, category}) :
5709 : : get_kvp_string_path (acc, {head}));
5710 : 45 : }
5711 : :
5712 : : GncGUID *
5713 : 2 : gnc_account_get_map_guid_entry (Account *acc, const char *head, const char *category)
5714 : : {
5715 : 2 : return category ?
5716 : 2 : get_kvp_guid_path (acc, {head, category}) :
5717 : 8 : get_kvp_guid_path (acc, {head});
5718 : 10 : }
5719 : :
5720 : : void
5721 : 1 : gnc_account_delete_map_entry (Account *acc, char *head, char *category,
5722 : : char *match_string, gboolean empty)
5723 : : {
5724 : 1 : if (acc != nullptr)
5725 : : {
5726 : 3 : std::vector<std::string> path {head};
5727 : 1 : if (category)
5728 : 0 : path.emplace_back (category);
5729 : 1 : if (match_string)
5730 : 0 : path.emplace_back (match_string);
5731 : :
5732 : 1 : if (qof_instance_has_path_slot (QOF_INSTANCE (acc), path))
5733 : : {
5734 : 1 : xaccAccountBeginEdit (acc);
5735 : 1 : if (empty)
5736 : 0 : qof_instance_slot_path_delete_if_empty (QOF_INSTANCE(acc), path);
5737 : : else
5738 : 1 : qof_instance_slot_path_delete (QOF_INSTANCE(acc), path);
5739 : 1 : PINFO("Account is '%s', head is '%s', category is '%s', match_string is'%s'",
5740 : : xaccAccountGetName (acc), head, category, match_string);
5741 : 1 : qof_instance_set_dirty (QOF_INSTANCE(acc));
5742 : 1 : xaccAccountCommitEdit (acc);
5743 : : }
5744 : 1 : }
5745 : 4 : }
5746 : :
5747 : : void
5748 : 0 : gnc_account_delete_all_bayes_maps (Account *acc)
5749 : : {
5750 : 0 : if (acc != nullptr)
5751 : : {
5752 : 0 : auto slots = qof_instance_get_slots_prefix (QOF_INSTANCE (acc), IMAP_FRAME_BAYES);
5753 : 0 : if (!slots.size()) return;
5754 : 0 : xaccAccountBeginEdit (acc);
5755 : 0 : for (auto const & entry : slots)
5756 : : {
5757 : 0 : qof_instance_slot_path_delete (QOF_INSTANCE (acc), {entry.first});
5758 : : }
5759 : 0 : qof_instance_set_dirty (QOF_INSTANCE(acc));
5760 : 0 : xaccAccountCommitEdit (acc);
5761 : 0 : }
5762 : 0 : }
5763 : :
5764 : : /* ================================================================ */
5765 : : /* QofObject function implementation and registration */
5766 : :
5767 : : static void
5768 : 1406 : destroy_all_child_accounts (Account *acc, gpointer data)
5769 : : {
5770 : 1406 : xaccAccountBeginEdit (acc);
5771 : 1406 : xaccAccountDestroy (acc);
5772 : 1406 : }
5773 : :
5774 : : static void
5775 : 173 : gnc_account_book_end(QofBook* book)
5776 : : {
5777 : 173 : Account *root_account = gnc_book_get_root_account (book);
5778 : : GList *accounts;
5779 : :
5780 : 173 : if (!root_account)
5781 : 70 : return;
5782 : :
5783 : 103 : accounts = gnc_account_get_descendants (root_account);
5784 : :
5785 : 103 : if (accounts)
5786 : : {
5787 : 101 : accounts = g_list_reverse (accounts);
5788 : 101 : g_list_foreach (accounts, (GFunc)destroy_all_child_accounts, nullptr);
5789 : 101 : g_list_free (accounts);
5790 : : }
5791 : 103 : xaccAccountBeginEdit (root_account);
5792 : 103 : xaccAccountDestroy (root_account);
5793 : : }
5794 : :
5795 : : #ifdef _MSC_VER
5796 : : /* MSVC compiler doesn't have C99 "designated initializers"
5797 : : * so we wrap them in a macro that is empty on MSVC. */
5798 : : # define DI(x) /* */
5799 : : #else
5800 : : # define DI(x) x
5801 : : #endif
5802 : : static QofObject account_object_def =
5803 : : {
5804 : : DI(.interface_version = ) QOF_OBJECT_VERSION,
5805 : : DI(.e_type = ) GNC_ID_ACCOUNT,
5806 : : DI(.type_label = ) "Account",
5807 : : DI(.create = ) (void*(*)(QofBook*)) xaccMallocAccount,
5808 : : DI(.book_begin = ) nullptr,
5809 : : DI(.book_end = ) gnc_account_book_end,
5810 : : DI(.is_dirty = ) qof_collection_is_dirty,
5811 : : DI(.mark_clean = ) qof_collection_mark_clean,
5812 : : DI(.foreach = ) qof_collection_foreach,
5813 : : DI(.printable = ) (const char * (*)(gpointer)) xaccAccountGetName,
5814 : : DI(.version_cmp = ) (int (*)(gpointer, gpointer)) qof_instance_version_cmp,
5815 : : };
5816 : :
5817 : 83 : gboolean xaccAccountRegister (void)
5818 : : {
5819 : : static QofParam params[] =
5820 : : {
5821 : : {
5822 : : ACCOUNT_NAME_, QOF_TYPE_STRING,
5823 : : (QofAccessFunc) xaccAccountGetName,
5824 : : (QofSetterFunc) xaccAccountSetName
5825 : : },
5826 : : {
5827 : : ACCOUNT_CODE_, QOF_TYPE_STRING,
5828 : : (QofAccessFunc) xaccAccountGetCode,
5829 : : (QofSetterFunc) xaccAccountSetCode
5830 : : },
5831 : : {
5832 : : ACCOUNT_DESCRIPTION_, QOF_TYPE_STRING,
5833 : : (QofAccessFunc) xaccAccountGetDescription,
5834 : : (QofSetterFunc) xaccAccountSetDescription
5835 : : },
5836 : : {
5837 : : ACCOUNT_COLOR_, QOF_TYPE_STRING,
5838 : : (QofAccessFunc) xaccAccountGetColor,
5839 : : (QofSetterFunc) xaccAccountSetColor
5840 : : },
5841 : : {
5842 : : ACCOUNT_FILTER_, QOF_TYPE_STRING,
5843 : : (QofAccessFunc) xaccAccountGetFilter,
5844 : : (QofSetterFunc) xaccAccountSetFilter
5845 : : },
5846 : : {
5847 : : ACCOUNT_SORT_ORDER_, QOF_TYPE_STRING,
5848 : : (QofAccessFunc) xaccAccountGetSortOrder,
5849 : : (QofSetterFunc) xaccAccountSetSortOrder
5850 : : },
5851 : : {
5852 : : ACCOUNT_SORT_REVERSED_, QOF_TYPE_BOOLEAN,
5853 : : (QofAccessFunc) xaccAccountGetSortReversed,
5854 : : (QofSetterFunc) xaccAccountSetSortReversed
5855 : : },
5856 : : {
5857 : : ACCOUNT_NOTES_, QOF_TYPE_STRING,
5858 : : (QofAccessFunc) xaccAccountGetNotes,
5859 : : (QofSetterFunc) xaccAccountSetNotes
5860 : : },
5861 : : {
5862 : : ACCOUNT_PRESENT_, QOF_TYPE_NUMERIC,
5863 : : (QofAccessFunc) xaccAccountGetPresentBalance, nullptr
5864 : : },
5865 : : {
5866 : : ACCOUNT_BALANCE_, QOF_TYPE_NUMERIC,
5867 : : (QofAccessFunc) xaccAccountGetBalance, nullptr
5868 : : },
5869 : : {
5870 : : ACCOUNT_CLEARED_, QOF_TYPE_NUMERIC,
5871 : : (QofAccessFunc) xaccAccountGetClearedBalance, nullptr
5872 : : },
5873 : : {
5874 : : ACCOUNT_RECONCILED_, QOF_TYPE_NUMERIC,
5875 : : (QofAccessFunc) xaccAccountGetReconciledBalance, nullptr
5876 : : },
5877 : : {
5878 : : ACCOUNT_TYPE_, QOF_TYPE_STRING,
5879 : : (QofAccessFunc) qofAccountGetTypeString,
5880 : : (QofSetterFunc) qofAccountSetType
5881 : : },
5882 : : {
5883 : : ACCOUNT_FUTURE_MINIMUM_, QOF_TYPE_NUMERIC,
5884 : : (QofAccessFunc) xaccAccountGetProjectedMinimumBalance, nullptr
5885 : : },
5886 : : {
5887 : : ACCOUNT_TAX_RELATED, QOF_TYPE_BOOLEAN,
5888 : : (QofAccessFunc) xaccAccountGetTaxRelated,
5889 : : (QofSetterFunc) xaccAccountSetTaxRelated
5890 : : },
5891 : : {
5892 : : ACCOUNT_OPENING_BALANCE_, QOF_TYPE_BOOLEAN,
5893 : : (QofAccessFunc) xaccAccountGetIsOpeningBalance,
5894 : : (QofSetterFunc) xaccAccountSetIsOpeningBalance
5895 : : },
5896 : : {
5897 : : ACCOUNT_SCU, QOF_TYPE_INT32,
5898 : : (QofAccessFunc) xaccAccountGetCommoditySCU,
5899 : : (QofSetterFunc) xaccAccountSetCommoditySCU
5900 : : },
5901 : : {
5902 : : ACCOUNT_NSCU, QOF_TYPE_BOOLEAN,
5903 : : (QofAccessFunc) xaccAccountGetNonStdSCU,
5904 : : (QofSetterFunc) xaccAccountSetNonStdSCU
5905 : : },
5906 : : {
5907 : : ACCOUNT_PARENT, GNC_ID_ACCOUNT,
5908 : : (QofAccessFunc) gnc_account_get_parent,
5909 : : (QofSetterFunc) qofAccountSetParent
5910 : : },
5911 : : {
5912 : : QOF_PARAM_BOOK, QOF_ID_BOOK,
5913 : : (QofAccessFunc) qof_instance_get_book, nullptr
5914 : : },
5915 : : {
5916 : : QOF_PARAM_GUID, QOF_TYPE_GUID,
5917 : : (QofAccessFunc) qof_instance_get_guid, nullptr
5918 : : },
5919 : : { nullptr },
5920 : : };
5921 : :
5922 : 83 : qof_class_register (GNC_ID_ACCOUNT, (QofSortFunc) qof_xaccAccountOrder, params);
5923 : :
5924 : 83 : return qof_object_register (&account_object_def);
5925 : : }
5926 : :
5927 : : /* ======================= UNIT TESTING ACCESS =======================
5928 : : * The following functions are for unit testing use only.
5929 : : */
5930 : : static AccountPrivate*
5931 : 11 : utest_account_get_private (Account *acc)
5932 : : {
5933 : 11 : return GET_PRIVATE (acc);
5934 : : }
5935 : :
5936 : : AccountTestFunctions*
5937 : 45 : _utest_account_fill_functions(void)
5938 : : {
5939 : 45 : AccountTestFunctions* func = g_new(AccountTestFunctions, 1);
5940 : :
5941 : 45 : func->get_private = utest_account_get_private;
5942 : 45 : func->coll_get_root_account = gnc_coll_get_root_account;
5943 : 45 : func->xaccFreeAccountChildren = xaccFreeAccountChildren;
5944 : 45 : func->xaccFreeAccount = xaccFreeAccount;
5945 : 45 : func->qofAccountSetParent = qofAccountSetParent;
5946 : 45 : func->gnc_account_lookup_by_full_name_helper =
5947 : : gnc_account_lookup_by_full_name_helper;
5948 : :
5949 : 45 : return func;
5950 : : }
5951 : : /* ======================= END OF FILE =========================== */
|