30 #include "gncTaxTableP.h" 32 #include "gnc-xml-helper.h" 34 #include "sixtp-utils.h" 35 #include "sixtp-parsers.h" 36 #include "sixtp-utils.h" 37 #include "sixtp-dom-parsers.h" 38 #include "sixtp-dom-generators.h" 41 #include "io-gncxml-gen.h" 44 #include "gnc-tax-table-xml-v2.h" 46 #define _GNC_MOD_NAME GNC_ID_TAXTABLE 48 static QofLogModule log_module = GNC_MOD_IO;
50 const gchar* taxtable_version_string =
"2.0.0";
53 #define gnc_taxtable_string "gnc:GncTaxTable" 54 #define taxtable_guid_string "taxtable:guid" 55 #define taxtable_name_string "taxtable:name" 56 #define taxtable_refcount_string "taxtable:refcount" 57 #define taxtable_invisible_string "taxtable:invisible" 58 #define taxtable_parent_string "taxtable:parent" 59 #define taxtable_child_string "taxtable:child" 60 #define taxtable_entries_string "taxtable:entries" 61 #define taxtable_slots_string "taxtable:slots" 63 #define gnc_taxtableentry_string "gnc:GncTaxTableEntry" 64 #define ttentry_account_string "tte:acct" 65 #define ttentry_type_string "tte:type" 66 #define ttentry_amount_string "tte:amount" 72 xmlAddChild (ptr, guid_to_dom_tree (tag,
83 ret = xmlNewNode (NULL, BAD_CAST gnc_taxtableentry_string);
85 account = gncTaxTableEntryGetAccount (entry);
87 xmlAddChild (ret, guid_to_dom_tree (ttentry_account_string,
90 amount = gncTaxTableEntryGetAmount (entry);
91 xmlAddChild (ret, gnc_numeric_to_dom_tree (ttentry_amount_string, &amount));
93 xmlAddChild (ret, text_to_dom_tree (ttentry_type_string,
94 gncAmountTypeToString (
95 gncTaxTableEntryGetType (entry))));
103 xmlNodePtr ret, entries;
106 ret = xmlNewNode (NULL, BAD_CAST gnc_taxtable_string);
107 xmlSetProp (ret, BAD_CAST
"version", BAD_CAST taxtable_version_string);
109 maybe_add_guid (ret, taxtable_guid_string,
table);
110 xmlAddChild (ret, text_to_dom_tree (taxtable_name_string,
111 gncTaxTableGetName (
table)));
113 xmlAddChild (ret, int_to_dom_tree (taxtable_refcount_string,
114 gncTaxTableGetRefcount (
table)));
115 xmlAddChild (ret, int_to_dom_tree (taxtable_invisible_string,
116 gncTaxTableGetInvisible (
table)));
120 maybe_add_guid (ret, taxtable_child_string, gncTaxTableGetChild (
table));
122 maybe_add_guid (ret, taxtable_parent_string, gncTaxTableGetParent (
table));
124 entries = xmlNewChild (ret, NULL, BAD_CAST taxtable_entries_string, NULL);
125 for (list = gncTaxTableGetEntries (
table); list; list = list->next)
128 xmlAddChild (entries, ttentry_dom_tree_create (entry));
132 xmlAddChild (ret, qof_instance_slots_to_dom_tree (taxtable_slots_string,
133 QOF_INSTANCE (
table)));
146 ttentry_acct_handler (xmlNodePtr node, gpointer
ttentry_pdata)
152 guid = dom_tree_to_guid (node);
153 g_return_val_if_fail (guid, FALSE);
156 g_return_val_if_fail (acc, FALSE);
158 gncTaxTableEntrySetAccount (pdata->ttentry, acc);
163 ttentry_type_handler (xmlNodePtr node, gpointer
ttentry_pdata)
170 str = dom_tree_to_text (node);
171 g_return_val_if_fail (str, FALSE);
173 ret = gncAmountStringToType (str, &type);
177 gncTaxTableEntrySetType (pdata->ttentry, type);
183 ttentry_amount_handler (xmlNodePtr node, gpointer
ttentry_pdata)
187 gncTaxTableEntrySetAmount (pdata->ttentry, dom_tree_to_gnc_numeric (node));
193 { ttentry_account_string, ttentry_acct_handler, 0, 0 },
194 { ttentry_type_string, ttentry_type_handler, 1, 0 },
195 { ttentry_amount_string, ttentry_amount_handler, 1, 0 },
200 dom_tree_to_ttentry (xmlNodePtr node, QofBook* book)
208 successful = dom_tree_generic_parse (node, ttentry_handlers_v2,
213 PERR (
"failed to parse tax table entry tree");
236 guid = dom_tree_to_guid (node);
237 g_return_val_if_fail (guid, FALSE);
238 table = gncTaxTableLookup (pdata->book, guid);
241 if (
table == pdata->table)
243 PINFO (
"found a self-referential parent/child; ignoring.\n");
249 table = gncTaxTableCreate (pdata->book);
250 gncTaxTableBeginEdit (
table);
251 gncTaxTableSetGUID (
table, guid);
252 gncTaxTableCommitEdit (
table);
255 g_return_val_if_fail (
table, FALSE);
256 func (pdata->table,
table);
268 guid = dom_tree_to_guid (node);
269 g_return_val_if_fail (guid, FALSE);
270 table = gncTaxTableLookup (pdata->book, guid);
273 gncTaxTableDestroy (pdata->table);
274 pdata->table =
table;
275 gncTaxTableBeginEdit (
table);
279 gncTaxTableSetGUID (pdata->table, guid);
291 char* txt = dom_tree_to_text (node);
292 g_return_val_if_fail (txt, FALSE);
294 gncTaxTableSetName (pdata->table, txt);
300 taxtable_refcount_handler (xmlNodePtr node, gpointer
taxtable_pdata)
305 dom_tree_to_integer (node, &val);
306 gncTaxTableSetRefcount (pdata->table, val);
311 taxtable_invisible_handler (xmlNodePtr node, gpointer
taxtable_pdata)
316 dom_tree_to_integer (node, &val);
318 gncTaxTableMakeInvisible (pdata->table);
323 taxtable_parent_handler (xmlNodePtr node, gpointer
taxtable_pdata)
326 return set_parent_child (node, pdata, gncTaxTableSetParent);
333 return set_parent_child (node, pdata, gncTaxTableSetChild);
337 taxtable_entries_handler (xmlNodePtr node, gpointer
taxtable_pdata)
342 g_return_val_if_fail (node, FALSE);
343 g_return_val_if_fail (node->xmlChildrenNode, FALSE);
345 for (mark = node->xmlChildrenNode; mark; mark = mark->next)
349 if (g_strcmp0 (
"text", (
char*)mark->name) == 0)
352 if (g_strcmp0 (gnc_taxtableentry_string, (
char*)mark->name))
355 entry = dom_tree_to_ttentry (mark, pdata->book);
358 gncTaxTableAddEntry (pdata->table, entry);
371 return dom_tree_create_instance_slots (node, QOF_INSTANCE (pdata->table));
376 { taxtable_guid_string, taxtable_guid_handler, 1, 0 },
377 { taxtable_name_string, taxtable_name_handler, 1, 0 },
378 { taxtable_refcount_string, taxtable_refcount_handler, 1, 0 },
379 { taxtable_invisible_string, taxtable_invisible_handler, 1, 0 },
380 { taxtable_parent_string, taxtable_parent_handler, 0, 0 },
381 { taxtable_child_string, taxtable_child_handler, 0, 0 },
382 { taxtable_entries_string, taxtable_entries_handler, 1, 0 },
383 { taxtable_slots_string, taxtable_slots_handler, 0, 0 },
388 dom_tree_to_taxtable (xmlNodePtr node, QofBook* book)
397 successful = dom_tree_generic_parse (node, taxtable_handlers_v2,
404 PERR (
"failed to parse tax table tree");
413 gnc_taxtable_end_handler (gpointer data_for_children,
414 GSList* data_from_children, GSList* sibling_data,
415 gpointer parent_data, gpointer global_data,
416 gpointer* result,
const gchar* tag)
419 xmlNodePtr tree = (xmlNodePtr)data_for_children;
420 gxpf_data* gdata = (gxpf_data*)global_data;
421 QofBook* book =
static_cast<decltype (book)
> (gdata->bookdata);
435 g_return_val_if_fail (tree, FALSE);
437 table = dom_tree_to_taxtable (tree, book);
440 gdata->cb (tag, gdata->parsedata,
table);
445 return table != NULL;
449 taxtable_sixtp_parser_create (
void)
451 return sixtp_dom_parser_new (gnc_taxtable_end_handler, NULL, NULL);
457 int* count =
static_cast<decltype (count)
> (count_p);
462 taxtable_get_count (QofBook* book)
470 xml_add_taxtable (
QofInstance* table_p, gpointer out_p)
474 FILE* out =
static_cast<decltype (out)
> (out_p);
479 node = taxtable_dom_tree_create (
table);
480 xmlElemDump (out, NULL, node);
482 if (ferror (out) || fprintf (out,
"\n") < 0)
487 taxtable_write (FILE* out, QofBook* book)
491 return ferror (out) == 0;
498 return (gncTaxTableGetParent (gncTaxTableGetParent (
table)) != NULL);
510 parent = gncTaxTableGetParent (temp);
513 gp = gncTaxTableGetParent (parent);
530 g_assert (gp == NULL);
538 taxtable_scrub_cb (
QofInstance* table_p, gpointer list_p)
541 GList** list =
static_cast<decltype (list)
> (list_p);
543 if (taxtable_is_grandchild (
table) || gncTaxTableGetEntries (
table) == NULL)
544 *list = g_list_prepend (*list,
table);
551 taxtable_scrub_entries (
QofInstance* entry_p, gpointer ht_p)
553 GHashTable* ht =
static_cast<decltype (ht)
> (ht_p);
554 GncEntry* entry = GNC_ENTRY (entry_p);
558 table = gncEntryGetInvTaxTable (entry);
561 if (taxtable_is_grandchild (
table))
565 PINFO (
"Fixing i-taxtable on entry %s\n", guidstr);
566 new_tt = taxtable_find_senior (
table);
567 gncEntryBeginEdit (entry);
568 gncEntrySetInvTaxTable (entry, new_tt);
569 gncEntryCommitEdit (entry);
574 count = GPOINTER_TO_INT (g_hash_table_lookup (ht,
table));
576 g_hash_table_insert (ht,
table, GINT_TO_POINTER (count));
580 table = gncEntryGetBillTaxTable (entry);
583 if (taxtable_is_grandchild (
table))
587 PINFO (
"Fixing b-taxtable on entry %s\n", guidstr);
588 new_tt = taxtable_find_senior (
table);
589 gncEntryBeginEdit (entry);
590 gncEntrySetBillTaxTable (entry, new_tt);
591 gncEntryCommitEdit (entry);
596 count = GPOINTER_TO_INT (g_hash_table_lookup (ht,
table));
598 g_hash_table_insert (ht,
table, GINT_TO_POINTER (count));
604 taxtable_scrub_cust (
QofInstance* cust_p, gpointer ht_p)
606 GHashTable* ht =
static_cast<decltype (ht)
> (ht_p);
611 table = gncCustomerGetTaxTable (cust);
614 count = GPOINTER_TO_INT (g_hash_table_lookup (ht,
table));
616 g_hash_table_insert (ht,
table, GINT_TO_POINTER (count));
621 taxtable_scrub_vendor (
QofInstance* vendor_p, gpointer ht_p)
623 GHashTable* ht =
static_cast<decltype (ht)
> (ht_p);
624 GncVendor* vendor = GNC_VENDOR (vendor_p);
628 table = gncVendorGetTaxTable (vendor);
631 count = GPOINTER_TO_INT (g_hash_table_lookup (ht,
table));
633 g_hash_table_insert (ht,
table, GINT_TO_POINTER (count));
638 taxtable_reset_refcount (gpointer key, gpointer value, gpointer notused)
641 gint32 count = GPOINTER_TO_INT (value);
643 if (count != gncTaxTableGetRefcount (
table) &&
644 !gncTaxTableGetInvisible (
table))
648 PWARN (
"Fixing refcount on taxtable %s (%" G_GINT64_FORMAT
" -> %d)\n",
649 guidstr, gncTaxTableGetRefcount (
table), count);
650 gncTaxTableSetRefcount (
table, count);
655 taxtable_scrub (QofBook* book)
660 GHashTable* ht = g_hash_table_new (g_direct_hash, g_direct_equal);
668 for (node = list; node; node = node->next)
671 table =
static_cast<decltype (
table)
> (node->data);
674 PINFO (
"deleting grandchild taxtable: %s\n", guidstr);
677 parent = gncTaxTableGetParent (
table);
678 gncTaxTableSetChild (parent, NULL);
681 gncTaxTableBeginEdit (
table);
682 gncTaxTableDestroy (
table);
686 g_hash_table_foreach (ht, taxtable_reset_refcount, NULL);
689 g_hash_table_destroy (ht);
693 taxtable_ns (FILE* out)
695 g_return_val_if_fail (out, FALSE);
697 gnc_xml2_write_namespace_decl (out,
"taxtable")
698 && gnc_xml2_write_namespace_decl (out,
"tte");
702 gnc_taxtable_xml_initialize (
void)
704 static GncXmlDataType_t be_data =
706 GNC_FILE_BACKEND_VERS,
708 taxtable_sixtp_parser_create,
716 gnc_xml_register_backend(be_data);
const GncGUID * qof_instance_get_guid(gconstpointer inst)
Return the GncGUID of this instance.
#define PINFO(format, args...)
Print an informational note.
gchar * guid_to_string_buff(const GncGUID *guid, gchar *str)
The guid_to_string_buff() routine puts a null-terminated string encoding of the id into the memory po...
#define PERR(format, args...)
Log a serious error.
#define PWARN(format, args...)
Log a warning.
void qof_object_foreach_sorted(QofIdTypeConst type_name, QofBook *book, QofInstanceForeachCB cb, gpointer user_data)
Invoke callback 'cb' on each instance in guid orted order.
api for GnuCash version 2 XML-based file format
#define GUID_ENCODING_LENGTH
Number of characters needed to encode a guid as a string not including the null terminator.
void qof_object_foreach(QofIdTypeConst type_name, QofBook *book, QofInstanceForeachCB cb, gpointer user_data)
Invoke the callback 'cb' on every instance ov a particular object type.
credit, discount and shipaddr are unique to GncCustomer id, name, notes, terms, addr, currency, taxtable, taxtable_override taxincluded, active and jobs are identical to ::GncVendor.
Business Entry Interface.
The type used to store guids in C.
GncAmountType
How to interpret the amount.
modtime is the internal date of the last modtime See libgnucash/engine/TaxTableBillTermImmutability.txt for an explanation of the following Code that handles refcount, parent, child, invisible and children is identical to that in ::GncBillTerm
Account * xaccAccountLookup(const GncGUID *guid, QofBook *book)
The xaccAccountLookup() subroutine will return the account associated with the given id...