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)
151 auto guid = dom_tree_to_guid (node);
152 g_return_val_if_fail (guid, FALSE);
154 g_return_val_if_fail (acc, FALSE);
156 gncTaxTableEntrySetAccount (pdata->ttentry, acc);
161 ttentry_type_handler (xmlNodePtr node, gpointer
ttentry_pdata)
167 if (gncAmountStringToType (str, &type))
168 gncTaxTableEntrySetType (tt, type);
170 return apply_xmlnode_text (tte_settype, pdata->ttentry, node);
174 ttentry_amount_handler (xmlNodePtr node, gpointer
ttentry_pdata)
178 gncTaxTableEntrySetAmount (pdata->ttentry, dom_tree_to_gnc_numeric (node));
184 { ttentry_account_string, ttentry_acct_handler, 0, 0 },
185 { ttentry_type_string, ttentry_type_handler, 1, 0 },
186 { ttentry_amount_string, ttentry_amount_handler, 1, 0 },
191 dom_tree_to_ttentry (xmlNodePtr node, QofBook* book)
199 successful = dom_tree_generic_parse (node, ttentry_handlers_v2,
204 PERR (
"failed to parse tax table entry tree");
226 auto guid = dom_tree_to_guid (node);
227 g_return_val_if_fail (guid, FALSE);
228 table = gncTaxTableLookup (pdata->book, &*guid);
231 if (
table == pdata->table)
233 PINFO (
"found a self-referential parent/child; ignoring.\n");
239 table = gncTaxTableCreate (pdata->book);
240 gncTaxTableBeginEdit (
table);
241 gncTaxTableSetGUID (
table, &*guid);
242 gncTaxTableCommitEdit (
table);
244 g_return_val_if_fail (
table, FALSE);
245 func (pdata->table,
table);
256 auto guid = dom_tree_to_guid (node);
257 g_return_val_if_fail (guid, FALSE);
258 table = gncTaxTableLookup (pdata->book, &*guid);
261 gncTaxTableDestroy (pdata->table);
262 pdata->table =
table;
263 gncTaxTableBeginEdit (
table);
267 gncTaxTableSetGUID (pdata->table, &*guid);
277 return apply_xmlnode_text (gncTaxTableSetName, pdata->table, node);
281 taxtable_refcount_handler (xmlNodePtr node, gpointer
taxtable_pdata)
286 dom_tree_to_integer (node, &val);
287 gncTaxTableSetRefcount (pdata->table, val);
292 taxtable_invisible_handler (xmlNodePtr node, gpointer
taxtable_pdata)
297 dom_tree_to_integer (node, &val);
299 gncTaxTableMakeInvisible (pdata->table);
304 taxtable_parent_handler (xmlNodePtr node, gpointer
taxtable_pdata)
307 return set_parent_child (node, pdata, gncTaxTableSetParent);
314 return set_parent_child (node, pdata, gncTaxTableSetChild);
318 taxtable_entries_handler (xmlNodePtr node, gpointer
taxtable_pdata)
323 g_return_val_if_fail (node, FALSE);
324 g_return_val_if_fail (node->xmlChildrenNode, FALSE);
326 for (mark = node->xmlChildrenNode; mark; mark = mark->next)
330 if (g_strcmp0 (
"text", (
char*)mark->name) == 0)
333 if (g_strcmp0 (gnc_taxtableentry_string, (
char*)mark->name))
336 entry = dom_tree_to_ttentry (mark, pdata->book);
339 gncTaxTableAddEntry (pdata->table, entry);
352 return dom_tree_create_instance_slots (node, QOF_INSTANCE (pdata->table));
357 { taxtable_guid_string, taxtable_guid_handler, 1, 0 },
358 { taxtable_name_string, taxtable_name_handler, 1, 0 },
359 { taxtable_refcount_string, taxtable_refcount_handler, 1, 0 },
360 { taxtable_invisible_string, taxtable_invisible_handler, 1, 0 },
361 { taxtable_parent_string, taxtable_parent_handler, 0, 0 },
362 { taxtable_child_string, taxtable_child_handler, 0, 0 },
363 { taxtable_entries_string, taxtable_entries_handler, 1, 0 },
364 { taxtable_slots_string, taxtable_slots_handler, 0, 0 },
369 dom_tree_to_taxtable (xmlNodePtr node, QofBook* book)
378 successful = dom_tree_generic_parse (node, taxtable_handlers_v2,
385 PERR (
"failed to parse tax table tree");
394 gnc_taxtable_end_handler (gpointer data_for_children,
395 GSList* data_from_children, GSList* sibling_data,
396 gpointer parent_data, gpointer global_data,
397 gpointer* result,
const gchar* tag)
400 xmlNodePtr tree = (xmlNodePtr)data_for_children;
401 gxpf_data* gdata = (gxpf_data*)global_data;
402 QofBook* book =
static_cast<decltype (book)
> (gdata->bookdata);
416 g_return_val_if_fail (tree, FALSE);
418 table = dom_tree_to_taxtable (tree, book);
421 gdata->cb (tag, gdata->parsedata,
table);
426 return table != NULL;
430 taxtable_sixtp_parser_create (
void)
432 return sixtp_dom_parser_new (gnc_taxtable_end_handler, NULL, NULL);
438 int* count =
static_cast<decltype (count)
> (count_p);
443 taxtable_get_count (QofBook* book)
451 xml_add_taxtable (
QofInstance* table_p, gpointer out_p)
455 FILE* out =
static_cast<decltype (out)
> (out_p);
460 node = taxtable_dom_tree_create (
table);
461 xmlElemDump (out, NULL, node);
463 if (ferror (out) || fprintf (out,
"\n") < 0)
468 taxtable_write (FILE* out, QofBook* book)
472 return ferror (out) == 0;
479 return (gncTaxTableGetParent (gncTaxTableGetParent (
table)) != NULL);
491 parent = gncTaxTableGetParent (temp);
494 gp = gncTaxTableGetParent (parent);
511 g_assert (gp == NULL);
519 taxtable_scrub_cb (
QofInstance* table_p, gpointer list_p)
522 GList** list =
static_cast<decltype (list)
> (list_p);
524 if (taxtable_is_grandchild (
table) || gncTaxTableGetEntries (
table) == NULL)
525 *list = g_list_prepend (*list,
table);
532 taxtable_scrub_entries (
QofInstance* entry_p, gpointer ht_p)
534 GHashTable* ht =
static_cast<decltype (ht)
> (ht_p);
535 GncEntry* entry = GNC_ENTRY (entry_p);
539 table = gncEntryGetInvTaxTable (entry);
542 if (taxtable_is_grandchild (
table))
546 PINFO (
"Fixing i-taxtable on entry %s\n", guidstr);
547 new_tt = taxtable_find_senior (
table);
548 gncEntryBeginEdit (entry);
549 gncEntrySetInvTaxTable (entry, new_tt);
550 gncEntryCommitEdit (entry);
555 count = GPOINTER_TO_INT (g_hash_table_lookup (ht,
table));
557 g_hash_table_insert (ht,
table, GINT_TO_POINTER (count));
561 table = gncEntryGetBillTaxTable (entry);
564 if (taxtable_is_grandchild (
table))
568 PINFO (
"Fixing b-taxtable on entry %s\n", guidstr);
569 new_tt = taxtable_find_senior (
table);
570 gncEntryBeginEdit (entry);
571 gncEntrySetBillTaxTable (entry, new_tt);
572 gncEntryCommitEdit (entry);
577 count = GPOINTER_TO_INT (g_hash_table_lookup (ht,
table));
579 g_hash_table_insert (ht,
table, GINT_TO_POINTER (count));
585 taxtable_scrub_cust (
QofInstance* cust_p, gpointer ht_p)
587 GHashTable* ht =
static_cast<decltype (ht)
> (ht_p);
592 table = gncCustomerGetTaxTable (cust);
595 count = GPOINTER_TO_INT (g_hash_table_lookup (ht,
table));
597 g_hash_table_insert (ht,
table, GINT_TO_POINTER (count));
602 taxtable_scrub_vendor (
QofInstance* vendor_p, gpointer ht_p)
604 GHashTable* ht =
static_cast<decltype (ht)
> (ht_p);
605 GncVendor* vendor = GNC_VENDOR (vendor_p);
609 table = gncVendorGetTaxTable (vendor);
612 count = GPOINTER_TO_INT (g_hash_table_lookup (ht,
table));
614 g_hash_table_insert (ht,
table, GINT_TO_POINTER (count));
619 taxtable_reset_refcount (gpointer key, gpointer value, gpointer notused)
622 gint32 count = GPOINTER_TO_INT (value);
624 if (count != gncTaxTableGetRefcount (
table) &&
625 !gncTaxTableGetInvisible (
table))
629 PWARN (
"Fixing refcount on taxtable %s (%" G_GINT64_FORMAT
" -> %d)\n",
630 guidstr, gncTaxTableGetRefcount (
table), count);
631 gncTaxTableSetRefcount (
table, count);
636 taxtable_scrub (QofBook* book)
641 GHashTable* ht = g_hash_table_new (g_direct_hash, g_direct_equal);
649 for (node = list; node; node = node->next)
652 table =
static_cast<decltype (
table)
> (node->data);
655 PINFO (
"deleting grandchild taxtable: %s\n", guidstr);
658 parent = gncTaxTableGetParent (
table);
659 gncTaxTableSetChild (parent, NULL);
662 gncTaxTableBeginEdit (
table);
663 gncTaxTableDestroy (
table);
667 g_hash_table_foreach (ht, taxtable_reset_refcount, NULL);
670 g_hash_table_destroy (ht);
674 taxtable_ns (FILE* out)
676 g_return_val_if_fail (out, FALSE);
678 gnc_xml2_write_namespace_decl (out,
"taxtable")
679 && gnc_xml2_write_namespace_decl (out,
"tte");
683 gnc_taxtable_xml_initialize (
void)
685 static GncXmlDataType_t be_data =
687 GNC_FILE_BACKEND_VERS,
689 taxtable_sixtp_parser_create,
697 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.
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...