GnuCash  5.6-150-g038405b370+
gnc-commodity.cpp
1 /********************************************************************
2  * gnc-commodity.c -- api for tradable commodities (incl. currency) *
3  * Copyright (C) 2000 Bill Gribble *
4  * Copyright (C) 2001,2003 Linas Vepstas <linas@linas.org> *
5  * Copyright (c) 2006 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 <glib.h>
29 #include <glib/gi18n.h>
30 #include <ctype.h>
31 #include <limits.h>
32 #include <string.h>
33 #include <stdio.h>
34 #include <stdlib.h>
35 #include <regex.h>
36 #include <qofinstance-p.h>
37 
38 #include "gnc-commodity.hpp"
39 #include "gnc-commodity.h"
40 #include "gnc-locale-utils.h"
41 #include "gnc-prefs.h"
42 #include "guid.h"
43 #include "qofinstance.h"
44 
45 #include <list>
46 #include <unordered_map>
47 
48 static QofLogModule log_module = GNC_MOD_COMMODITY;
49 
50 /* Parts per unit is nominal, i.e. number of 'partname' units in
51  * a 'unitname' unit. fraction is transactional, i.e. how many
52  * of the smallest-transactional-units of the currency are there
53  * in a 'unitname' unit. */
54 
55 enum
56 {
57  PROP_0,
58  PROP_NAMESPACE, /* Table */
59  PROP_FULL_NAME, /* Table */
60  PROP_MNEMONIC, /* Table */
61  PROP_PRINTNAME, /* Constructed */
62  PROP_CUSIP, /* Table */
63  PROP_FRACTION, /* Table */
64  PROP_UNIQUE_NAME, /* Constructed */
65  PROP_QUOTE_FLAG, /* Table */
66  PROP_QUOTE_SOURCE, /* Table */
67  PROP_QUOTE_TZ, /* Table */
68 };
69 
71 {
72  QofInstance inst;
73 };
74 
75 typedef struct gnc_commodityPrivate
76 {
77  gnc_commodity_namespace *name_space;
78 
79  const char *fullname;
80  const char *mnemonic;
81  char *printname;
82  const char *cusip; /* CUSIP or other identifying code */
83  int fraction;
84  char *unique_name;
85  char *user_symbol;
86 
87  gboolean quote_flag; /* user wants price quotes */
88  gnc_quote_source *quote_source; /* current/old source of quotes */
89  const char *quote_tz;
90 
91  /* the number of accounts using this commodity - this field is not
92  * persisted */
93  int usage_count;
94 
95  /* the default display_symbol, set in iso-4217-currencies at start-up */
96  const char *default_symbol;
98 
99 #define GET_PRIVATE(o) \
100  ((gnc_commodityPrivate*)gnc_commodity_get_instance_private((gnc_commodity*)o))
101 
103 {
104  QofInstanceClass parent_class;
105 };
106 
107 static void commodity_free(gnc_commodity * cm);
108 static void gnc_commodity_set_default_symbol(gnc_commodity *, const char *);
109 
111 {
112  QofInstance inst;
113 
114  const gchar *name;
115  gboolean iso4217;
116  GHashTable * cm_table;
117  GList * cm_list;
118 };
119 
121 {
122  QofInstanceClass parent_class;
123 };
124 
126 {
127  GHashTable * ns_table;
128  GList * ns_list;
129 };
130 
131 static const std::unordered_map<std::string,std::string> gnc_new_iso_codes =
132 {
133  {"RUR", "RUB"}, /* Russian Ruble: RUR through 1997-12, RUB from 1998-01 onwards; see bug #393185 */
134  {"PLZ", "PLN"}, /* Polish Zloty */
135  {"UAG", "UAH"}, /* Ukraine Hryvnia */
136  {"NIS", "ILS"}, /* New Israeli Shekel: The informal abbreviation may be "NIS", but
137  its iso-4217 is clearly ILS and only this! Incorrectly changed
138  due to bug#152755 (Nov 2004) and changed back again by bug#492417
139  (Oct 2008). */
140  {"MXP", "MXN"}, /* Mexican (Nuevo) Peso */
141  {"TRL", "TRY"}, /* New Turkish Lira: changed 2005 */
142 
143  /* Only add currencies to this table when the old currency no longer
144  * exists in the file iso-4217-currencies.xml */
145 };
146 
147 static std::string fq_version;
148 
150 {
151 private:
152  gboolean m_supported;
153  QuoteSourceType m_type;
154  std::string m_user_name; /* User friendly name incl. region code*/
155  std::string m_internal_name; /* Name used internally and by finance::quote. */
156 public:
157  bool get_supported () const { return m_supported; }
158  void set_supported (bool supported) { m_supported = supported; }
159  QuoteSourceType get_type () const { return m_type; }
160  const char* get_user_name () const { return m_user_name.c_str(); }
161  const char* get_internal_name () const { return m_internal_name.c_str(); }
162  gnc_quote_source_s (gboolean supported, QuoteSourceType type,
163  const char* username, const char* int_name)
164  : m_supported{supported}
165  , m_type{type}
166  , m_user_name{username ? username : ""}
167  , m_internal_name{int_name ? int_name: ""} { };
168 };
169 
170 using QuoteSourceList = std::list<gnc_quote_source>;
171 
172 /* To update the following lists scan
173  * from github.com/finance-quote/finance-quote
174  * in lib/Finance/Quote/ all *.pm for "methods"
175  * because many of them have more than one -
176  * ideally after each release of them.
177  *
178  * Apply changes here also to the FQ appendix of help.
179  */
180 static QuoteSourceList currency_quote_sources =
181 {
182  { true, SOURCE_CURRENCY, "Currency", "currency" }
183 };
184 
185 /* The single quote method is usually the module name, but
186  * sometimes it gets the suffix "_direct"
187  * and the failover method is without suffix.
188  */
189 static QuoteSourceList single_quote_sources =
190 {
191  { false, SOURCE_SINGLE, NC_("FQ Source", "Alphavantage"), "alphavantage" },
192  { false, SOURCE_SINGLE, NC_("FQ Source", "Amsterdam Euronext eXchange, NL"), "aex" },
193  { false, SOURCE_SINGLE, NC_("FQ Source", "Association of Mutual Funds in India"), "amfiindia" },
194  { false, SOURCE_SINGLE, NC_("FQ Source", "Athens Exchange Group, GR"), "asegr" },
195  { false, SOURCE_SINGLE, NC_("FQ Source", "Australian Stock Exchange, AU"), "asx" },
196  { false, SOURCE_SINGLE, NC_("FQ Source", "Italian Stock Exchange, IT"), "borsa_italiana" },
197  { false, SOURCE_SINGLE, NC_("FQ Source", "BSE India, IN"), "bseindia" },
198  { false, SOURCE_SINGLE, NC_("FQ Source", "Bucharest Stock Exchange, RO"), "bvb" },
199  { false, SOURCE_SINGLE, NC_("FQ Source", "China Merchants Bank, CN"), "cmbchina" },
200  { false, SOURCE_SINGLE, NC_("FQ Source", "Colombo Stock Exchange, LK"), "cse" },
201  { false, SOURCE_SINGLE, NC_("FQ Source", "comdirect, DE"), "comdirect" },
202  { false, SOURCE_SINGLE, NC_("FQ Source", "Consors Bank, DE"), "consorsbank" },
203  { false, SOURCE_SINGLE, NC_("FQ Source", "Deka Investments, DE"), "deka" },
204  { false, SOURCE_SINGLE, NC_("FQ Source", "Financial Times Funds service, GB"), "ftfunds" },
205  { false, SOURCE_SINGLE, NC_("FQ Source", "Finanzpartner, DE"), "finanzpartner" },
206  { false, SOURCE_SINGLE, NC_("FQ Source", "FondsWeb, DE"), "fondsweb" },
207  { false, SOURCE_SINGLE, NC_("FQ Source", "GoldMoney precious metals"), "goldmoney" },
208  { false, SOURCE_SINGLE, NC_("FQ Source", "Google Web, US Stocks"), "googleweb" },
209  { false, SOURCE_SINGLE, NC_("FQ Source", "Market Watch"), "marketwatch" },
210  { false, SOURCE_SINGLE, NC_("FQ Source", "Morningstar, CH"), "morningstarch" },
211  { false, SOURCE_SINGLE, NC_("FQ Source", "Morningstar, GB"), "morningstaruk" },
212  { false, SOURCE_SINGLE, NC_("FQ Source", "Morningstar, JP"), "morningstarjp" },
213  { false, SOURCE_SINGLE, NC_("FQ Source", "Motley Fool"), "fool" },
214  { false, SOURCE_SINGLE, NC_("FQ Source", "New Zealand stock eXchange, NZ"), "nzx" },
215  { false, SOURCE_SINGLE, NC_("FQ Source", "NSE (National Stock Exchange), IN"), "nseindia" },
216  { false, SOURCE_SINGLE, NC_("FQ Source", "OnVista, DE"), "onvista"},
217  { false, SOURCE_SINGLE, NC_("FQ Source", "Paris Stock Exchange/Boursorama, FR"), "bourso" },
218  { false, SOURCE_SINGLE, NC_("FQ Source", "S-Investor, DE"), "sinvestor"},
219  { false, SOURCE_SINGLE, NC_("FQ Source", "Sharenet, ZA"), "za" },
220  { false, SOURCE_SINGLE, NC_("FQ Source", "SIX Swiss Exchange shares, CH"), "six" },
221  { false, SOURCE_SINGLE, NC_("FQ Source", "StockData"), "stockdata" },
222  { false, SOURCE_SINGLE, NC_("FQ Source", "Stooq, PL"), "stooq" },
223  { false, SOURCE_SINGLE, NC_("FQ Source", "Tesouro Direto bonds, BR"), "tesouro_direto" },
224  { false, SOURCE_SINGLE, NC_("FQ Source", "Toronto Stock eXchange, CA"), "tsx" },
225  { false, SOURCE_SINGLE, NC_("FQ Source", "Tradegate, DE"), "tradegate" },
226  { false, SOURCE_SINGLE, NC_("FQ Source", "Treasury Direct bonds, US"), "treasurydirect" },
227  { false, SOURCE_SINGLE, NC_("FQ Source", "Twelve Data"), "twelvedata" },
228  { false, SOURCE_SINGLE, NC_("FQ Source", "Union Investment, DE"), "unionfunds" },
229  { false, SOURCE_SINGLE, NC_("FQ Source", "US Govt. Thrift Savings Plan"), "tsp" },
230  { false, SOURCE_SINGLE, NC_("FQ Source", "XETRA, DE"), "xetra" },
231  { false, SOURCE_SINGLE, NC_("FQ Source", "Yahoo as JSON"), "yahoo_json" },
232  { false, SOURCE_SINGLE, NC_("FQ Source", "Yahoo Web"), "yahooweb" },
233  { false, SOURCE_SINGLE, NC_("FQ Source", "YH Finance (FinanceAPI)"), "financeapi" },
234 };
235 
236 // Finance::Quote defines these as failover methods
237 static QuoteSourceList multiple_quote_sources =
238 {
239  { false, SOURCE_MULTI, NC_("FQ Source", "Canada (Alphavantage, TMX)"), "canada" },
240  { false, SOURCE_MULTI, NC_("FQ Source", "Europe (ASEGR, Bourso, …)"), "europe" },
241  { false, SOURCE_MULTI, NC_("FQ Source", "India (BSEIndia, NSEIndia)"), "india"},
242  { false, SOURCE_MULTI, NC_("FQ Source", "Nasdaq (Alphavantage, FinanceAPI, …)"), "nasdaq" },
243  { false, SOURCE_MULTI, NC_("FQ Source", "NYSE (Alphavantage, FinanceAPI, …)"), "nyse" },
244  { false, SOURCE_MULTI, NC_("FQ Source", "U.K. Funds (FTfunds, MorningstarUK)"), "ukfunds" },
245  { false, SOURCE_MULTI, NC_("FQ Source", "USA (Alphavantage, FinanceAPI, …)"), "usa" },
246 };
247 
248 static QuoteSourceList new_quote_sources;
249 
250 // cannot use map or unordered_map because order must be preserved
251 static const std::vector<std::pair<QuoteSourceType,QuoteSourceList&>> quote_sources_map =
252  {
253  { SOURCE_CURRENCY, currency_quote_sources },
254  { SOURCE_SINGLE, single_quote_sources },
255  { SOURCE_MULTI, multiple_quote_sources },
256  { SOURCE_UNKNOWN, new_quote_sources }
257  };
258 
259 /********************************************************************
260  * gnc_quote_source_fq_installed
261  *
262  * This function indicates whether or not the Finance::Quote module
263  * is installed on a users computer.
264  ********************************************************************/
265 gboolean
267 {
268  return (!fq_version.empty());
269 }
270 
271 
272 /********************************************************************
273  * gnc_quote_source_fq_version
274  *
275  * This function the version of the Finance::Quote module installed
276  * on a user's computer or nullptr if no installation is found.
277  ********************************************************************/
278 const char*
280 {
281  return fq_version.c_str();
282 }
283 
284 static QuoteSourceList&
285 get_quote_source_from_type (QuoteSourceType type)
286 {
287  auto quote_sources_it = std::find_if (quote_sources_map.begin(), quote_sources_map.end(),
288  [type] (const auto& qs) { return type == qs.first; });
289 
290  if (quote_sources_it != quote_sources_map.end())
291  return quote_sources_it->second;
292 
293  PWARN ("Invalid Quote Source %d, returning new_quote_sources", type);
294  return new_quote_sources;
295 }
296 
297 /********************************************************************
298  * gnc_quote_source_num_entries
299  *
300  * Return the number of entries for a given type of price source.
301  ********************************************************************/
303 {
304  auto source{get_quote_source_from_type(type)};
305  return std::distance(source.begin(), source.end());
306 }
307 
308 
309 
310 /********************************************************************
311  * gnc_quote_source_add_new
312  *
313  * Add a new price source. Called when unknown source names are found
314  * either in the F::Q installation (a newly available source) or in
315  * the user's data file (a source that has vanished but needs to be
316  * tracked.)
317  ********************************************************************/
318 gnc_quote_source *
319 gnc_quote_source_add_new (const char *source_name, gboolean supported)
320 {
321  DEBUG("Creating new source %s", (!source_name ? "(null)" : source_name));
322  /* This name can be changed if/when support for this price source is
323  * integrated into gnucash. */
324  /* This name is permanent and must be kept the same if/when support
325  * for this price source is integrated into gnucash (i.e. for a
326  * nice user name). */
327  return &new_quote_sources.emplace_back (supported, SOURCE_UNKNOWN, source_name, source_name);
328 }
329 
330 /********************************************************************
331  * gnc_quote_source_lookup_by_xxx
332  *
333  * Lookup a price source data structure based upon various criteria.
334  ********************************************************************/
335 gnc_quote_source *
337 {
338  ENTER("type/index is %d/%d", type, index);
339  auto& sources = get_quote_source_from_type (type);
340  if ((size_t) index < sources.size())
341  {
342  auto it = std::next(sources.begin(), index);
343  LEAVE("found %s", it->get_user_name());
344  return &*it;
345  }
346 
347  LEAVE("not found");
348  return nullptr;
349 }
350 
351 gnc_quote_source *
353 {
354  if (!name || !*name)
355  return nullptr;
356 
357  for (const auto& [_, sources] : quote_sources_map)
358  {
359  auto source_it = std::find_if (sources.begin(), sources.end(),
360  [name] (const auto& qs)
361  { return (g_strcmp0(name, qs.get_internal_name()) == 0); });
362  if (source_it != sources.end())
363  return &(*source_it);
364  }
365 
366  DEBUG("gnc_quote_source_lookup_by_internal: Unknown source %s", name);
367  return nullptr;
368 }
369 
370 /********************************************************************
371  * gnc_quote_source_get_xxx
372  *
373  * Accessor functions - get functions only. There are no set functions.
374  ********************************************************************/
376 gnc_quote_source_get_type (const gnc_quote_source *source)
377 {
378  ENTER("%p", source);
379  if (!source)
380  {
381  LEAVE("bad source");
382  return SOURCE_SINGLE;
383  }
384 
385  LEAVE("type is %d", source->get_type());
386  return source->get_type();
387 }
388 
389 gint
390 gnc_quote_source_get_index (const gnc_quote_source *source)
391 {
392  if (!source)
393  {
394  PWARN ("bad source");
395  return 0;
396  }
397 
398  auto& sources = get_quote_source_from_type (source->get_type());
399  auto is_source = [&source](const auto& findif_source)
400  { return &findif_source == source; };
401 
402  auto iter = std::find_if (sources.begin(), sources.end(), is_source);
403  if (iter != sources.end())
404  return std::distance (sources.begin(), iter);
405 
406  PWARN ("couldn't locate source");
407  return 0;
408 }
409 
410 gboolean
411 gnc_quote_source_get_supported (const gnc_quote_source *source)
412 {
413  ENTER("%p", source);
414  if (!source)
415  {
416  LEAVE("bad source");
417  return FALSE;
418  }
419 
420  LEAVE("%s supported", source && source->get_supported() ? "" : "not ");
421  return source->get_supported();
422 }
423 
424 const char *
425 gnc_quote_source_get_user_name (const gnc_quote_source *source)
426 {
427  ENTER("%p", source);
428  if (!source)
429  {
430  LEAVE("bad source");
431  return nullptr;
432  }
433  LEAVE("user name %s", source->get_user_name());
434  return source->get_user_name();
435 }
436 
437 const char *
438 gnc_quote_source_get_internal_name (const gnc_quote_source *source)
439 {
440  ENTER("%p", source);
441  if (!source)
442  {
443  LEAVE("bad source");
444  return nullptr;
445  }
446  LEAVE("internal name %s", source->get_internal_name());
447  return source->get_internal_name();
448 }
449 
450 
451 /********************************************************************
452  * gnc_quote_source_set_fq_installed
453  *
454  * Update gnucash internal tables on what Finance::Quote sources are
455  * installed.
456  ********************************************************************/
457 void
458 gnc_quote_source_set_fq_installed (const char* version_string,
459  const std::vector<std::string>& sources_list)
460 {
461  ENTER(" ");
462 
463  if (sources_list.empty())
464  return;
465 
466  if (version_string)
467  fq_version = version_string;
468  else
469  fq_version.clear();
470 
471  for (const auto& source_name_str : sources_list)
472  {
473  auto source_name = source_name_str.c_str();
474  auto source = gnc_quote_source_lookup_by_internal(source_name);
475 
476  if (source)
477  {
478  DEBUG("Found source %s: %s", source_name, source->get_user_name());
479  source->set_supported (true);
480  continue;
481  }
482 
483  gnc_quote_source_add_new(source_name, TRUE);
484  }
485  LEAVE(" ");
486 }
487 
488 /********************************************************************
489  * QoF Helpers
490  ********************************************************************/
491 
492 void
493 gnc_commodity_begin_edit (gnc_commodity *cm)
494 {
495  qof_begin_edit(&cm->inst);
496 }
497 
498 static void commit_err (QofInstance *inst, QofBackendError errcode)
499 {
500  PERR ("Failed to commit: %d", errcode);
501  gnc_engine_signal_commit_error( errcode );
502 }
503 
504 static void noop (QofInstance *inst) {}
505 
506 static void
507 comm_free(QofInstance* inst)
508 {
509  commodity_free( GNC_COMMODITY(inst) );
510 }
511 
512 void
513 gnc_commodity_commit_edit (gnc_commodity *cm)
514 {
515  if (!qof_commit_edit (QOF_INSTANCE(cm))) return;
516  qof_commit_edit_part2 (&cm->inst, commit_err, noop, comm_free);
517 }
518 
519 /********************************************************************
520  * gnc_commodity_new
521  ********************************************************************/
522 
523 static void
524 mark_commodity_dirty (gnc_commodity *cm)
525 {
526  qof_instance_set_dirty(&cm->inst);
527  qof_event_gen (&cm->inst, QOF_EVENT_MODIFY, nullptr);
528 }
529 
530 static void
531 reset_printname(gnc_commodityPrivate *priv)
532 {
533  g_free(priv->printname);
534  priv->printname = g_strdup_printf("%s (%s)",
535  priv->mnemonic ? priv->mnemonic : "",
536  priv->fullname ? priv->fullname : "");
537 }
538 
539 static void
540 reset_unique_name(gnc_commodityPrivate *priv)
541 {
542  gnc_commodity_namespace *ns;
543 
544  g_free(priv->unique_name);
545  ns = priv->name_space;
546  priv->unique_name = g_strdup_printf("%s::%s",
547  ns ? ns->name : "",
548  priv->mnemonic ? priv->mnemonic : "");
549 }
550 
551 /* GObject Initialization */
552 G_DEFINE_TYPE_WITH_PRIVATE(gnc_commodity, gnc_commodity, QOF_TYPE_INSTANCE)
553 
554 static void
555 gnc_commodity_init(gnc_commodity* com)
556 {
557  gnc_commodityPrivate* priv;
558 
559  priv = GET_PRIVATE(com);
560 
561  priv->name_space = nullptr;
562  priv->fullname = CACHE_INSERT("");
563  priv->mnemonic = CACHE_INSERT("");
564  priv->cusip = CACHE_INSERT("");
565  priv->fraction = 10000;
566  priv->quote_flag = 0;
567  priv->quote_source = nullptr;
568  priv->quote_tz = CACHE_INSERT("");
569 
570  reset_printname(priv);
571  reset_unique_name(priv);
572 }
573 
574 static void
575 gnc_commodity_dispose(GObject *comp)
576 {
577  G_OBJECT_CLASS(gnc_commodity_parent_class)->dispose(comp);
578 }
579 
580 static void
581 gnc_commodity_finalize(GObject* comp)
582 {
583  G_OBJECT_CLASS(gnc_commodity_parent_class)->finalize(comp);
584 }
585 /* Note that g_value_set_object() refs the object, as does
586  * g_object_get(). But g_object_get() only unrefs once when it disgorges
587  * the object, leaving an unbalanced ref, which leaks. So instead of
588  * using g_value_set_object(), use g_value_take_object() which doesn't
589  * ref the object when used in get_property().
590  */
591 static void
592 gnc_commodity_get_property (GObject *object,
593  guint prop_id,
594  GValue *value,
595  GParamSpec *pspec)
596 {
597  gnc_commodity *commodity;
598  gnc_commodityPrivate* priv;
599 
600  g_return_if_fail(GNC_IS_COMMODITY(object));
601 
602  commodity = GNC_COMMODITY(object);
603  priv = GET_PRIVATE(commodity);
604  switch (prop_id)
605  {
606  case PROP_NAMESPACE:
607  g_value_take_object(value, priv->name_space);
608  break;
609  case PROP_FULL_NAME:
610  g_value_set_string(value, priv->fullname);
611  break;
612  case PROP_MNEMONIC:
613  g_value_set_string(value, priv->mnemonic);
614  break;
615  case PROP_PRINTNAME:
616  g_value_set_string(value, priv->printname);
617  break;
618  case PROP_CUSIP:
619  g_value_set_string(value, priv->cusip);
620  break;
621  case PROP_FRACTION:
622  g_value_set_int(value, priv->fraction);
623  break;
624  case PROP_UNIQUE_NAME:
625  g_value_set_string(value, priv->unique_name);
626  break;
627  case PROP_QUOTE_FLAG:
628  g_value_set_boolean(value, priv->quote_flag);
629  break;
630  case PROP_QUOTE_SOURCE:
631  g_value_set_pointer(value, priv->quote_source);
632  break;
633  case PROP_QUOTE_TZ:
634  g_value_set_string(value, priv->quote_tz);
635  break;
636  default:
637  G_OBJECT_WARN_INVALID_PROPERTY_ID(object, prop_id, pspec);
638  break;
639  }
640 }
641 
642 static void
643 gnc_commodity_set_property (GObject *object,
644  guint prop_id,
645  const GValue *value,
646  GParamSpec *pspec)
647 {
648  gnc_commodity *commodity;
649 
650  g_return_if_fail(GNC_IS_COMMODITY(object));
651 
652  commodity = GNC_COMMODITY(object);
653  g_assert (qof_instance_get_editlevel(commodity));
654 
655  switch (prop_id)
656  {
657  case PROP_NAMESPACE:
658  gnc_commodity_set_namespace(commodity, static_cast<const char*>(g_value_get_object(value)));
659  break;
660  case PROP_FULL_NAME:
661  gnc_commodity_set_fullname(commodity, g_value_get_string(value));
662  break;
663  case PROP_MNEMONIC:
664  gnc_commodity_set_mnemonic(commodity, g_value_get_string(value));
665  break;
666  case PROP_CUSIP:
667  gnc_commodity_set_cusip(commodity, g_value_get_string(value));
668  break;
669  case PROP_FRACTION:
670  gnc_commodity_set_fraction(commodity, g_value_get_int(value));
671  break;
672  case PROP_QUOTE_FLAG:
673  gnc_commodity_set_quote_flag(commodity, g_value_get_boolean(value));
674  break;
675  case PROP_QUOTE_SOURCE:
676  gnc_commodity_set_quote_source(commodity, static_cast<gnc_quote_source*>(g_value_get_pointer(value)));
677  break;
678  case PROP_QUOTE_TZ:
679  gnc_commodity_set_quote_tz(commodity, g_value_get_string(value));
680  break;
681  default:
682  G_OBJECT_WARN_INVALID_PROPERTY_ID(object, prop_id, pspec);
683  break;
684  }
685 }
686 static void
687 gnc_commodity_class_init(struct _GncCommodityClass* klass)
688 {
689  GObjectClass *gobject_class = G_OBJECT_CLASS(klass);
690 
691  gobject_class->dispose = gnc_commodity_dispose;
692  gobject_class->finalize = gnc_commodity_finalize;
693  gobject_class->set_property = gnc_commodity_set_property;
694  gobject_class->get_property = gnc_commodity_get_property;
695 
696  g_object_class_install_property(gobject_class,
697  PROP_NAMESPACE,
698  g_param_spec_object ("namespace",
699  "Namespace",
700  "The namespace field denotes the "
701  "namespace for this commodity, either "
702  "a currency or symbol from a quote source.",
703  GNC_TYPE_COMMODITY_NAMESPACE,
704  G_PARAM_READWRITE));
705  g_object_class_install_property(gobject_class,
706  PROP_FULL_NAME,
707  g_param_spec_string ("fullname",
708  "Full Commodity Name",
709  "The fullname is the official full name of"
710  "the currency.",
711  nullptr,
712  G_PARAM_READWRITE));
713  g_object_class_install_property(gobject_class,
714  PROP_MNEMONIC,
715  g_param_spec_string ("mnemonic",
716  "Commodity Mnemonic",
717  "The mnemonic is the official abbreviated"
718  "designation for the currency.",
719  nullptr,
720  G_PARAM_READWRITE));
721  g_object_class_install_property(gobject_class,
722  PROP_PRINTNAME,
723  g_param_spec_string ("printname",
724  "Commodity Print Name",
725  "Printable form of the commodity name.",
726  nullptr,
727  G_PARAM_READABLE));
728  g_object_class_install_property(gobject_class,
729  PROP_CUSIP,
730  g_param_spec_string ("cusip",
731  "Commodity CUSIP Code",
732  "?????",
733  nullptr,
734  G_PARAM_READWRITE));
735  g_object_class_install_property(gobject_class,
736  PROP_FRACTION,
737  g_param_spec_int ("fraction",
738  "Fraction",
739  "The fraction is the number of sub-units that "
740  "the basic commodity can be divided into.",
741  1,
743  1,
744  G_PARAM_READWRITE));
745  g_object_class_install_property(gobject_class,
746  PROP_UNIQUE_NAME,
747  g_param_spec_string ("unique-name",
748  "Commodity Unique Name",
749  "Unique form of the commodity name which combines "
750  "the namespace name and the commodity name.",
751  nullptr,
752  G_PARAM_READABLE));
753  g_object_class_install_property(gobject_class,
754  PROP_QUOTE_FLAG,
755  g_param_spec_boolean ("quote_flag",
756  "Quote Flag",
757  "TRUE if prices are to be downloaded for this "
758  "commodity from a quote source.",
759  FALSE,
760  G_PARAM_READWRITE));
761  g_object_class_install_property(gobject_class,
762  PROP_QUOTE_SOURCE,
763  g_param_spec_pointer("quote-source",
764  "Quote Source",
765  "The quote source from which prices are downloaded.",
766  G_PARAM_READWRITE));
767  g_object_class_install_property(gobject_class,
768  PROP_QUOTE_TZ,
769  g_param_spec_string ("quote-tz",
770  "Commodity Quote Timezone",
771  "?????",
772  nullptr,
773  G_PARAM_READWRITE));
774 }
775 
776 gnc_commodity *
777 gnc_commodity_new(QofBook *book, const char * fullname,
778  const char * name_space, const char * mnemonic,
779  const char * cusip, int fraction)
780 {
781  auto retval = GNC_COMMODITY(g_object_new(GNC_TYPE_COMMODITY, nullptr));
782 
783  qof_instance_init_data (&retval->inst, GNC_ID_COMMODITY, book);
784  gnc_commodity_begin_edit(retval);
785 
786  if ( name_space != nullptr )
787  {
788  /* Prevent setting anything except template in namespace template. */
789  if (g_strcmp0 (name_space, GNC_COMMODITY_NS_TEMPLATE) == 0 &&
790  g_strcmp0 (mnemonic, "template") != 0)
791  {
792  PWARN("Converting commodity %s from namespace template to "
793  "namespace User", mnemonic);
794  name_space = "User";
795  }
796  gnc_commodity_set_namespace(retval, name_space);
797  if (gnc_commodity_namespace_is_iso(name_space))
798  {
801  }
802  }
803  gnc_commodity_set_fullname(retval, fullname);
804  gnc_commodity_set_mnemonic(retval, mnemonic);
805  gnc_commodity_set_cusip(retval, cusip);
806  gnc_commodity_set_fraction(retval, fraction);
807  mark_commodity_dirty (retval);
808  gnc_commodity_commit_edit(retval);
809 
810  qof_event_gen (&retval->inst, QOF_EVENT_CREATE, nullptr);
811 
812  return retval;
813 }
814 
815 
816 /********************************************************************
817  * gnc_commodity_destroy
818  ********************************************************************/
819 
820 static void
821 commodity_free(gnc_commodity * cm)
822 {
823  QofBook *book;
824  gnc_commodity_table *table;
825  gnc_commodityPrivate* priv;
826 
827  if (!cm) return;
828 
829  book = qof_instance_get_book(&cm->inst);
832  priv = GET_PRIVATE(cm);
833 
834  qof_event_gen (&cm->inst, QOF_EVENT_DESTROY, nullptr);
835 
836  /* Set at creation */
837  CACHE_REMOVE (priv->fullname);
838  CACHE_REMOVE (priv->cusip);
839  CACHE_REMOVE (priv->mnemonic);
840  CACHE_REMOVE (priv->quote_tz);
841  priv->name_space = nullptr;
842 
843  /* Set through accessor functions */
844  priv->quote_source = nullptr;
845 
846  /* Automatically generated */
847  g_free(priv->printname);
848  priv->printname = nullptr;
849 
850  g_free(priv->unique_name);
851  priv->unique_name = nullptr;
852 
853 #ifdef ACCOUNTS_CLEANED_UP
854  /* Account objects are not actually cleaned up when a book is closed (in fact
855  * a memory leak), but commodities are, so in currently this warning gets hit
856  * quite frequently. Disable the check until cleaning up of accounts objects
857  * on close is implemented. */
858  if (priv->usage_count != 0)
859  {
860  PWARN("Destroying commodity (%p) with non-zero usage_count (%d).", cm,
861  priv->usage_count);
862  }
863 #endif
864 
865  /* qof_instance_release (&cm->inst); */
866  g_object_unref(cm);
867 }
868 
869 void
870 gnc_commodity_destroy(gnc_commodity * cm)
871 {
872  gnc_commodity_begin_edit(cm);
873  qof_instance_set_destroying(cm, TRUE);
874  gnc_commodity_commit_edit(cm);
875 }
876 
877 void
878 gnc_commodity_copy(gnc_commodity * dest, const gnc_commodity *src)
879 {
880  gnc_commodityPrivate* src_priv = GET_PRIVATE(src);
881  gnc_commodityPrivate* dest_priv = GET_PRIVATE(dest);
882 
883  gnc_commodity_set_fullname (dest, src_priv->fullname);
884  gnc_commodity_set_mnemonic (dest, src_priv->mnemonic);
885  dest_priv->name_space = src_priv->name_space;
886  gnc_commodity_set_fraction (dest, src_priv->fraction);
887  gnc_commodity_set_cusip (dest, src_priv->cusip);
888  gnc_commodity_set_quote_flag (dest, src_priv->quote_flag);
890  gnc_commodity_set_quote_tz (dest, src_priv->quote_tz);
891  qof_instance_copy_kvp (QOF_INSTANCE (dest), QOF_INSTANCE (src));
892 }
893 
894 gnc_commodity *
895 gnc_commodity_clone(const gnc_commodity *src, QofBook *dest_book)
896 {
897  gnc_commodityPrivate* src_priv;
898  gnc_commodityPrivate* dest_priv;
899 
900  auto dest = GNC_COMMODITY (g_object_new(GNC_TYPE_COMMODITY, nullptr));
901  qof_instance_init_data (&dest->inst, GNC_ID_COMMODITY, dest_book);
902  src_priv = GET_PRIVATE(src);
903  dest_priv = GET_PRIVATE(dest);
904 
905  dest_priv->fullname = CACHE_INSERT(src_priv->fullname);
906  dest_priv->mnemonic = CACHE_INSERT(src_priv->mnemonic);
907  dest_priv->cusip = CACHE_INSERT(src_priv->cusip);
908  dest_priv->quote_tz = CACHE_INSERT(src_priv->quote_tz);
909 
910  dest_priv->name_space = src_priv->name_space;
911 
912  dest_priv->fraction = src_priv->fraction;
913  dest_priv->quote_flag = src_priv->quote_flag;
914 
916 
917  qof_instance_copy_kvp (QOF_INSTANCE (dest), QOF_INSTANCE (src));
918 
919  reset_printname(dest_priv);
920  reset_unique_name(dest_priv);
921 
922  return dest;
923 }
924 
925 /********************************************************************
926  * gnc_commodity_get_mnemonic
927  ********************************************************************/
928 
929 const char *
930 gnc_commodity_get_mnemonic(const gnc_commodity * cm)
931 {
932  if (!cm) return nullptr;
933  return GET_PRIVATE(cm)->mnemonic;
934 }
935 
936 /********************************************************************
937  * gnc_commodity_get_printname
938  ********************************************************************/
939 
940 const char *
941 gnc_commodity_get_printname(const gnc_commodity * cm)
942 {
943  if (!cm) return nullptr;
944  return GET_PRIVATE(cm)->printname;
945 }
946 
947 
948 /********************************************************************
949  * gnc_commodity_get_namespace
950  ********************************************************************/
951 
952 const char *
953 gnc_commodity_get_namespace(const gnc_commodity * cm)
954 {
955  if (!cm) return nullptr;
956  return gnc_commodity_namespace_get_name(GET_PRIVATE(cm)->name_space);
957 }
958 
959 gnc_commodity_namespace *
960 gnc_commodity_get_namespace_ds(const gnc_commodity * cm)
961 {
962  if (!cm) return nullptr;
963  return GET_PRIVATE(cm)->name_space;
964 }
965 
966 /********************************************************************
967  * gnc_commodity_get_fullname
968  ********************************************************************/
969 
970 const char *
971 gnc_commodity_get_fullname(const gnc_commodity * cm)
972 {
973  if (!cm) return nullptr;
974  return GET_PRIVATE(cm)->fullname;
975 }
976 
977 
978 /********************************************************************
979  * gnc_commodity_get_unique_name
980  ********************************************************************/
981 
982 const char *
983 gnc_commodity_get_unique_name(const gnc_commodity * cm)
984 {
985  if (!cm) return nullptr;
986  return GET_PRIVATE(cm)->unique_name;
987 }
988 
989 
990 /********************************************************************
991  * gnc_commodity_get_cusip
992  ********************************************************************/
993 
994 const char *
995 gnc_commodity_get_cusip(const gnc_commodity * cm)
996 {
997  if (!cm) return nullptr;
998  return GET_PRIVATE(cm)->cusip;
999 }
1000 
1001 /********************************************************************
1002  * gnc_commodity_get_fraction
1003  ********************************************************************/
1004 
1005 int
1006 gnc_commodity_get_fraction(const gnc_commodity * cm)
1007 {
1008  if (!cm) return 0;
1009  return GET_PRIVATE(cm)->fraction;
1010 }
1011 
1012 /********************************************************************
1013  * gnc_commodity_get_auto_quote_control_flag
1014  ********************************************************************/
1015 
1016 gboolean
1017 gnc_commodity_get_auto_quote_control_flag(const gnc_commodity *cm)
1018 {
1019  if (!cm) return FALSE;
1020  auto str{qof_instance_get_path_kvp<const char*> (QOF_INSTANCE (cm), {"auto_quote_control"})};
1021  return !str || g_strcmp0 (*str, "false");
1022 }
1023 
1024 /********************************************************************
1025  * gnc_commodity_get_quote_flag
1026  ********************************************************************/
1027 
1028 gboolean
1029 gnc_commodity_get_quote_flag(const gnc_commodity *cm)
1030 {
1031  if (!cm) return FALSE;
1032  return (GET_PRIVATE(cm)->quote_flag);
1033 }
1034 
1035 /********************************************************************
1036  * gnc_commodity_get_quote_source
1037  ********************************************************************/
1038 
1039 gnc_quote_source*
1040 gnc_commodity_get_quote_source(const gnc_commodity *cm)
1041 {
1042  gnc_commodityPrivate* priv;
1043 
1044  if (!cm) return nullptr;
1045  priv = GET_PRIVATE(cm);
1046  if (!priv->quote_source && gnc_commodity_is_iso(cm))
1047  return &currency_quote_sources.front();
1048  return priv->quote_source;
1049 }
1050 
1051 gnc_quote_source*
1052 gnc_commodity_get_default_quote_source(const gnc_commodity *cm)
1053 {
1054  if (cm && gnc_commodity_is_iso(cm))
1055  return &currency_quote_sources.front();
1056  /* Should make this a user option at some point. */
1057  return gnc_quote_source_lookup_by_internal("alphavantage");
1058 }
1059 
1060 /********************************************************************
1061  * gnc_commodity_get_quote_tz
1062  ********************************************************************/
1063 
1064 const char*
1065 gnc_commodity_get_quote_tz(const gnc_commodity *cm)
1066 {
1067  if (!cm) return nullptr;
1068  return GET_PRIVATE(cm)->quote_tz;
1069 }
1070 
1071 /********************************************************************
1072  * gnc_commodity_get_user_symbol
1073  ********************************************************************/
1074 const char*
1075 gnc_commodity_get_user_symbol(const gnc_commodity *cm)
1076 {
1077  g_return_val_if_fail (GNC_IS_COMMODITY (cm), nullptr);
1078 
1079  auto sym{qof_instance_get_path_kvp<const char*> (QOF_INSTANCE(cm), {"user_symbol"})};
1080  return sym ? *sym : nullptr;
1081 }
1082 
1083 /********************************************************************
1084  * gnc_commodity_get_default_symbol
1085  *******************************************************************/
1086 const char*
1087 gnc_commodity_get_default_symbol(const gnc_commodity *cm)
1088 {
1089  if (!cm) return nullptr;
1090  return GET_PRIVATE(cm)->default_symbol;
1091 }
1092 
1093 /********************************************************************
1094  * gnc_commodity_get_nice_symbol
1095  *******************************************************************/
1096 const char*
1097 gnc_commodity_get_nice_symbol (const gnc_commodity *cm)
1098 {
1099  const char *nice_symbol;
1100  struct lconv *lc;
1101  if (!cm) return nullptr;
1102 
1103  nice_symbol = gnc_commodity_get_user_symbol(cm);
1104  if (nice_symbol && *nice_symbol)
1105  return nice_symbol;
1106 
1107  lc = gnc_localeconv();
1108  nice_symbol = lc->currency_symbol;
1109  if (!g_strcmp0(gnc_commodity_get_mnemonic(cm), lc->int_curr_symbol))
1110  return nice_symbol;
1111 
1112  nice_symbol = gnc_commodity_get_default_symbol(cm);
1113  if (nice_symbol && *nice_symbol)
1114  return nice_symbol;
1115 
1116  return gnc_commodity_get_mnemonic(cm);
1117 }
1118 
1119 /********************************************************************
1120  * gnc_commodity_set_mnemonic
1121  ********************************************************************/
1122 
1123 void
1124 gnc_commodity_set_mnemonic(gnc_commodity * cm, const char * mnemonic)
1125 {
1126  gnc_commodityPrivate* priv;
1127 
1128  if (!cm) return;
1129  priv = GET_PRIVATE(cm);
1130  if (priv->mnemonic == mnemonic) return;
1131 
1132  gnc_commodity_begin_edit(cm);
1133  CACHE_REMOVE (priv->mnemonic);
1134  priv->mnemonic = CACHE_INSERT(mnemonic);
1135 
1136  mark_commodity_dirty (cm);
1137  reset_printname(priv);
1138  reset_unique_name(priv);
1139  gnc_commodity_commit_edit(cm);
1140 }
1141 
1142 /********************************************************************
1143  * gnc_commodity_set_namespace
1144  ********************************************************************/
1145 
1146 void
1147 gnc_commodity_set_namespace(gnc_commodity * cm, const char * name_space)
1148 {
1149  QofBook *book;
1150  gnc_commodity_table *table;
1151  gnc_commodity_namespace *nsp;
1152  gnc_commodityPrivate* priv;
1153 
1154  if (!cm) return;
1155  priv = GET_PRIVATE(cm);
1156  book = qof_instance_get_book (&cm->inst);
1158  nsp = gnc_commodity_table_add_namespace(table, name_space, book);
1159  if (priv->name_space == nsp)
1160  return;
1161 
1162  gnc_commodity_begin_edit(cm);
1163  priv->name_space = nsp;
1164  if (nsp->iso4217)
1165  priv->quote_source = gnc_quote_source_lookup_by_internal("currency");
1166  mark_commodity_dirty(cm);
1167  reset_printname(priv);
1168  reset_unique_name(priv);
1169  gnc_commodity_commit_edit(cm);
1170 }
1171 
1172 /********************************************************************
1173  * gnc_commodity_set_fullname
1174  ********************************************************************/
1175 
1176 void
1177 gnc_commodity_set_fullname(gnc_commodity * cm, const char * fullname)
1178 {
1179  gnc_commodityPrivate* priv;
1180 
1181  if (!cm) return;
1182  priv = GET_PRIVATE(cm);
1183  if (priv->fullname == fullname) return;
1184 
1185  CACHE_REMOVE (priv->fullname);
1186  priv->fullname = CACHE_INSERT (fullname);
1187 
1188  gnc_commodity_begin_edit(cm);
1189  mark_commodity_dirty(cm);
1190  reset_printname(priv);
1191  gnc_commodity_commit_edit(cm);
1192 }
1193 
1194 /********************************************************************
1195  * gnc_commodity_set_cusip
1196  ********************************************************************/
1197 
1198 void
1199 gnc_commodity_set_cusip(gnc_commodity * cm,
1200  const char * cusip)
1201 {
1202  gnc_commodityPrivate* priv;
1203 
1204  if (!cm) return;
1205 
1206  priv = GET_PRIVATE(cm);
1207  if (priv->cusip == cusip) return;
1208 
1209  gnc_commodity_begin_edit(cm);
1210  CACHE_REMOVE (priv->cusip);
1211  priv->cusip = CACHE_INSERT (cusip);
1212  mark_commodity_dirty(cm);
1213  gnc_commodity_commit_edit(cm);
1214 }
1215 
1216 /********************************************************************
1217  * gnc_commodity_set_fraction
1218  ********************************************************************/
1219 
1220 void
1221 gnc_commodity_set_fraction(gnc_commodity * cm, int fraction)
1222 {
1223  if (!cm) return;
1224  gnc_commodity_begin_edit(cm);
1225  GET_PRIVATE(cm)->fraction = fraction;
1226  mark_commodity_dirty(cm);
1227  gnc_commodity_commit_edit(cm);
1228 }
1229 
1230 /********************************************************************
1231  * gnc_commodity_set_auto_quote_control_flag
1232  ********************************************************************/
1233 
1234 void
1235 gnc_commodity_set_auto_quote_control_flag(gnc_commodity *cm,
1236  const gboolean flag)
1237 {
1238  ENTER ("(cm=%p, flag=%d)", cm, flag);
1239 
1240  if (!cm)
1241  {
1242  LEAVE("");
1243  return;
1244  }
1245  gnc_commodity_begin_edit(cm);
1246  auto val = flag ? std::nullopt : std::make_optional<const char*>(g_strdup("false"));
1247  qof_instance_set_path_kvp<const char*> (QOF_INSTANCE (cm), val, {"auto_quote_control"});
1248  mark_commodity_dirty(cm);
1249  gnc_commodity_commit_edit(cm);
1250  LEAVE("");
1251 }
1252 
1253 /********************************************************************
1254  * gnc_commodity_user_set_quote_flag
1255  ********************************************************************/
1256 
1257 void
1258 gnc_commodity_user_set_quote_flag(gnc_commodity *cm, const gboolean flag)
1259 {
1260  gnc_commodityPrivate* priv;
1261 
1262  ENTER ("(cm=%p, flag=%d)", cm, flag);
1263 
1264  if (!cm)
1265  {
1266  LEAVE("");
1267  return;
1268  }
1269 
1270  priv = GET_PRIVATE(cm);
1271  gnc_commodity_begin_edit(cm);
1272  gnc_commodity_set_quote_flag(cm, flag);
1273  if (gnc_commodity_is_iso(cm))
1274  {
1275  /* For currencies, disable auto quote control if the quote flag is being
1276  * changed from its default value and enable it if the quote flag is being
1277  * reset to its default value. The defaults for the quote flag are
1278  * disabled if no accounts are using the currency, and true otherwise.
1279  * Thus enable auto quote control if flag is FALSE and there are not any
1280  * accounts using this currency OR flag is TRUE and there are accounts
1281  * using this currency; otherwise disable auto quote control */
1282  gnc_commodity_set_auto_quote_control_flag(cm,
1283  (!flag && (priv->usage_count == 0)) || (flag && (priv->usage_count != 0)));
1284  }
1285  gnc_commodity_commit_edit(cm);
1286  LEAVE("");
1287 }
1288 
1289 /********************************************************************
1290  * gnc_commodity_set_quote_flag
1291  ********************************************************************/
1292 
1293 void
1294 gnc_commodity_set_quote_flag(gnc_commodity *cm, const gboolean flag)
1295 {
1296  ENTER ("(cm=%p, flag=%d)", cm, flag);
1297 
1298  if (!cm) return;
1299  gnc_commodity_begin_edit(cm);
1300  GET_PRIVATE(cm)->quote_flag = flag;
1301  mark_commodity_dirty(cm);
1302  gnc_commodity_commit_edit(cm);
1303  LEAVE(" ");
1304 }
1305 
1306 /********************************************************************
1307  * gnc_commodity_set_quote_source
1308  ********************************************************************/
1309 
1310 void
1311 gnc_commodity_set_quote_source(gnc_commodity *cm, gnc_quote_source *src)
1312 {
1313  ENTER ("(cm=%p, src=%p(%s))", cm, src, src ? src->get_internal_name() : "unknown");
1314 
1315  if (!cm) return;
1316  gnc_commodity_begin_edit(cm);
1317  GET_PRIVATE(cm)->quote_source = src;
1318  mark_commodity_dirty(cm);
1319  gnc_commodity_commit_edit(cm);
1320  LEAVE(" ");
1321 }
1322 
1323 /********************************************************************
1324  * gnc_commodity_set_quote_tz
1325  ********************************************************************/
1326 
1327 void
1328 gnc_commodity_set_quote_tz(gnc_commodity *cm, const char *tz)
1329 {
1330  gnc_commodityPrivate* priv;
1331 
1332  if (!cm) return;
1333 
1334  ENTER ("(cm=%p, tz=%s)", cm, tz ? tz : "(null)");
1335 
1336  priv = GET_PRIVATE(cm);
1337 
1338  if (tz == priv->quote_tz)
1339  {
1340  LEAVE("Already correct TZ");
1341  return;
1342  }
1343 
1344  gnc_commodity_begin_edit(cm);
1345  CACHE_REMOVE (priv->quote_tz);
1346  priv->quote_tz = CACHE_INSERT (tz);
1347  mark_commodity_dirty(cm);
1348  gnc_commodity_commit_edit(cm);
1349  LEAVE(" ");
1350 }
1351 
1352 /********************************************************************
1353  * gnc_commodity_set_user_symbol
1354  ********************************************************************/
1355 
1356 void
1357 gnc_commodity_set_user_symbol(gnc_commodity * cm, const char * user_symbol)
1358 {
1359  struct lconv *lc;
1360 
1361  if (!cm) return;
1362 
1363  ENTER ("(cm=%p, symbol=%s)", cm, user_symbol ? user_symbol : "(null)");
1364 
1365  lc = gnc_localeconv();
1366  if (!user_symbol || !*user_symbol)
1367  user_symbol = nullptr;
1368  else if (!g_strcmp0(lc->int_curr_symbol, gnc_commodity_get_mnemonic(cm)) &&
1369  !g_strcmp0(lc->currency_symbol, user_symbol))
1370  /* if the user gives the ISO symbol for the locale currency or the
1371  * default symbol, actually remove the user symbol */
1372  user_symbol = nullptr;
1373  else if (!g_strcmp0(user_symbol, gnc_commodity_get_default_symbol(cm)))
1374  user_symbol = nullptr;
1375 
1376  gnc_commodity_begin_edit (cm);
1377 
1378  auto val = user_symbol ? std::make_optional<const char*>(g_strdup(user_symbol)) : std::nullopt;
1379  qof_instance_set_path_kvp<const char*> (QOF_INSTANCE(cm), val, {"user_symbol"});
1380 
1381  mark_commodity_dirty(cm);
1382  gnc_commodity_commit_edit(cm);
1383 
1384  LEAVE(" ");
1385 }
1386 
1387 /********************************************************************
1388  * gnc_commodity_set_default_symbol
1389  * Not made visible in gnc-commodity.h, it is only called from
1390  * iso-4217-currencies.c at startup.
1391  ********************************************************************/
1392 void
1393 gnc_commodity_set_default_symbol(gnc_commodity * cm,
1394  const char * default_symbol)
1395 {
1396  GET_PRIVATE(cm)->default_symbol = default_symbol;
1397 }
1398 
1399 /********************************************************************
1400  * gnc_commodity_increment_usage_count
1401  ********************************************************************/
1402 
1403 void
1405 {
1406  gnc_commodityPrivate* priv;
1407 
1408  ENTER("(cm=%p)", cm);
1409 
1410  if (!cm)
1411  {
1412  LEAVE("");
1413  return;
1414  }
1415 
1416  priv = GET_PRIVATE(cm);
1417 
1418  if ((priv->usage_count == 0) && !priv->quote_flag
1419  && gnc_commodity_get_auto_quote_control_flag(cm)
1420  && gnc_commodity_is_iso(cm))
1421  {
1422  /* compatibility hack - Gnucash 1.8 gets currency quotes when a
1423  non-default currency is assigned to an account. */
1424  gnc_commodity_begin_edit(cm);
1425  gnc_commodity_set_quote_flag(cm, TRUE);
1427  gnc_commodity_get_default_quote_source(cm));
1428  gnc_commodity_commit_edit(cm);
1429  }
1430  priv->usage_count++;
1431  LEAVE("(usage_count=%d)", priv->usage_count);
1432 }
1433 
1434 /********************************************************************
1435  * gnc_commodity_decrement_usage_count
1436  ********************************************************************/
1437 
1438 void
1440 {
1441  gnc_commodityPrivate* priv;
1442 
1443  ENTER("(cm=%p)", cm);
1444 
1445  if (!cm)
1446  {
1447  LEAVE("");
1448  return;
1449  }
1450 
1451  priv = GET_PRIVATE(cm);
1452 
1453  if (priv->usage_count == 0)
1454  {
1455  PWARN("usage_count already zero");
1456  LEAVE("");
1457  return;
1458  }
1459 
1460  priv->usage_count--;
1461  if ((priv->usage_count == 0) && priv->quote_flag
1462  && gnc_commodity_get_auto_quote_control_flag(cm)
1463  && gnc_commodity_is_iso(cm))
1464  {
1465  /* if this is a currency with auto quote control enabled and no more
1466  * accounts reference this currency, disable quote retrieval */
1467  gnc_commodity_set_quote_flag(cm, FALSE);
1468  }
1469  LEAVE("(usage_count=%d)", priv->usage_count);
1470 }
1471 
1472 /********************************************************************\
1473 \********************************************************************/
1474 
1475 
1476 /********************************************************************
1477  * gnc_commodity_equiv
1478  * are two commodities the same?
1479  ********************************************************************/
1480 
1481 gboolean
1482 gnc_commodity_equiv(const gnc_commodity * a, const gnc_commodity * b)
1483 {
1484  gnc_commodityPrivate* priv_a;
1485  gnc_commodityPrivate* priv_b;
1486 
1487  if (a == b) return TRUE;
1488  if (!a || !b) return FALSE;
1489 
1490  priv_a = GET_PRIVATE(a);
1491  priv_b = GET_PRIVATE(b);
1492  if (priv_a->name_space != priv_b->name_space) return FALSE;
1493  if (g_strcmp0(priv_a->mnemonic, priv_b->mnemonic) != 0) return FALSE;
1494 
1495  return TRUE;
1496 }
1497 
1498 gboolean
1499 gnc_commodity_equal(const gnc_commodity * a, const gnc_commodity * b)
1500 {
1501  return gnc_commodity_compare(a, b) == 0;
1502 }
1503 
1504 // Used as a sorting callback for deleting old prices, so it needs to be
1505 // stable but doesn't need to be in any particular order sensible to humans.
1506 int gnc_commodity_compare(const gnc_commodity * a, const gnc_commodity * b)
1507 {
1508  if (a == b) return 0;
1509  if (a && !b) return 1;
1510  if (b && !a) return -1;
1511  return qof_instance_guid_compare(a, b);
1512 }
1513 
1514 // Used as a callback to g_list_find_custom, it should return 0
1515 // when the commodities match.
1516 int gnc_commodity_compare_void(const void * a, const void * b)
1517 {
1518  return gnc_commodity_compare(GNC_COMMODITY (a), GNC_COMMODITY (b));
1519 }
1520 
1521 /************************************************************
1522  * Namespace functions *
1523  ************************************************************/
1524 const char *
1525 gnc_commodity_namespace_get_name (const gnc_commodity_namespace *ns)
1526 {
1527  if (ns == nullptr)
1528  return nullptr;
1529  return ns->name;
1530 }
1531 
1532 const char *
1533 gnc_commodity_namespace_get_gui_name (const gnc_commodity_namespace *ns)
1534 {
1535  if (ns == nullptr)
1536  return nullptr;
1537  if (g_strcmp0 (ns->name, GNC_COMMODITY_NS_CURRENCY) == 0)
1538  return GNC_COMMODITY_NS_ISO_GUI;
1539  return ns->name;
1540 }
1541 
1542 GList *
1543 gnc_commodity_namespace_get_commodity_list(const gnc_commodity_namespace *name_space)
1544 {
1545  if (!name_space)
1546  return nullptr;
1547 
1548  return g_list_copy (name_space->cm_list);
1549 }
1550 
1551 gboolean
1552 gnc_commodity_namespace_is_iso(const char *name_space)
1553 {
1554  return ((g_strcmp0(name_space, GNC_COMMODITY_NS_ISO) == 0) ||
1555  (g_strcmp0(name_space, GNC_COMMODITY_NS_CURRENCY) == 0));
1556 }
1557 
1558 static const gchar *
1559 gnc_commodity_table_map_namespace(const char * name_space)
1560 {
1561  if (g_strcmp0(name_space, GNC_COMMODITY_NS_ISO) == 0)
1562  return GNC_COMMODITY_NS_CURRENCY;
1563  return name_space;
1564 }
1565 
1566 /********************************************************************
1567  * gnc_commodity_table_new
1568  * make a new commodity table
1569  ********************************************************************/
1570 
1571 gnc_commodity_table *
1573 {
1574  gnc_commodity_table * retval = g_new0(gnc_commodity_table, 1);
1575  retval->ns_table = g_hash_table_new(&g_str_hash, &g_str_equal);
1576  retval->ns_list = nullptr;
1577  return retval;
1578 }
1579 
1580 /********************************************************************
1581  * book anchor functions
1582  ********************************************************************/
1583 
1584 gnc_commodity_table *
1586 {
1587  if (!book) return nullptr;
1588  return static_cast<gnc_commodity_table*>(qof_book_get_data (book, GNC_COMMODITY_TABLE));
1589 }
1590 
1591 gnc_commodity *
1592 gnc_commodity_obtain_twin (const gnc_commodity *from, QofBook *book)
1593 {
1594  gnc_commodity *twin;
1595  const char * ucom;
1596  gnc_commodity_table * comtbl;
1597 
1598  if (!from) return nullptr;
1599  comtbl = gnc_commodity_table_get_table (book);
1600  if (!comtbl) return nullptr;
1601 
1602  ucom = gnc_commodity_get_unique_name (from);
1603  twin = gnc_commodity_table_lookup_unique (comtbl, ucom);
1604  if (!twin)
1605  {
1606  twin = gnc_commodity_clone (from, book);
1607  twin = gnc_commodity_table_insert (comtbl, twin);
1608  }
1609  return twin;
1610 }
1611 
1612 /********************************************************************
1613  * gnc_commodity_table_get_size
1614  * get the size of the commodity table
1615  ********************************************************************/
1616 
1617 static void
1618 count_coms(gpointer key, gpointer value, gpointer user_data)
1619 {
1620  GHashTable *tbl = ((gnc_commodity_namespace*)value)->cm_table;
1621  guint *count = (guint*)user_data;
1622 
1623  if (g_strcmp0((char*)key, GNC_COMMODITY_NS_CURRENCY) == 0)
1624  {
1625  /* don't count default commodities */
1626  return;
1627  }
1628 
1629  if (!value) return;
1630 
1631  *count += g_hash_table_size(tbl);
1632 }
1633 
1634 guint
1635 gnc_commodity_table_get_size(const gnc_commodity_table* tbl)
1636 {
1637  guint count = 0;
1638  g_return_val_if_fail(tbl, 0);
1639  g_return_val_if_fail(tbl->ns_table, 0);
1640 
1641  g_hash_table_foreach(tbl->ns_table, count_coms, (gpointer)&count);
1642 
1643  return count;
1644 }
1645 
1646 /********************************************************************
1647  * gnc_commodity_table_lookup
1648  * locate a commodity by namespace and mnemonic.
1649  ********************************************************************/
1650 
1651 gnc_commodity *
1652 gnc_commodity_table_lookup(const gnc_commodity_table * table,
1653  const char * name_space, const char * mnemonic)
1654 {
1655  gnc_commodity_namespace * nsp = nullptr;
1656 
1657  if (!table || !name_space || !mnemonic) return nullptr;
1658 
1659  nsp = gnc_commodity_table_find_namespace(table, name_space);
1660 
1661  if (nsp)
1662  {
1663  /*
1664  * Backward compatibility support for currencies that have
1665  * recently changed.
1666  */
1667  if (nsp->iso4217)
1668  {
1669  auto it = gnc_new_iso_codes.find (mnemonic);
1670  if (it != gnc_new_iso_codes.end())
1671  mnemonic = it->second.c_str();
1672  }
1673  return GNC_COMMODITY(g_hash_table_lookup(nsp->cm_table, (gpointer)mnemonic));
1674  }
1675  else
1676  {
1677  return nullptr;
1678  }
1679 }
1680 
1681 /********************************************************************
1682  * gnc_commodity_table_lookup
1683  * locate a commodity by unique name.
1684  ********************************************************************/
1685 
1686 gnc_commodity *
1687 gnc_commodity_table_lookup_unique(const gnc_commodity_table *table,
1688  const char * unique_name)
1689 {
1690  char *name_space;
1691  char *mnemonic;
1692  gnc_commodity *commodity;
1693 
1694  if (!table || !unique_name) return nullptr;
1695 
1696  name_space = g_strdup (unique_name);
1697  mnemonic = strstr (name_space, "::");
1698  if (!mnemonic)
1699  {
1700  g_free (name_space);
1701  return nullptr;
1702  }
1703 
1704  *mnemonic = '\0';
1705  mnemonic += 2;
1706 
1707  commodity = gnc_commodity_table_lookup (table, name_space, mnemonic);
1708 
1709  g_free (name_space);
1710 
1711  return commodity;
1712 }
1713 
1714 /********************************************************************
1715  * gnc_commodity_table_find_full
1716  * locate a commodity by namespace and printable name
1717  ********************************************************************/
1718 
1719 gnc_commodity *
1720 gnc_commodity_table_find_full(const gnc_commodity_table * table,
1721  const char * name_space,
1722  const char * fullname)
1723 {
1724  gnc_commodity * retval = nullptr;
1725  GList * all;
1726  GList * iterator;
1727 
1728  if (!fullname || (fullname[0] == '\0'))
1729  return nullptr;
1730 
1731  all = gnc_commodity_table_get_commodities(table, name_space);
1732 
1733  for (iterator = all; iterator; iterator = iterator->next)
1734  {
1735  auto commodity = GNC_COMMODITY (iterator->data);
1736  if (!strcmp(fullname,
1737  gnc_commodity_get_printname(commodity)))
1738  {
1739  retval = commodity;
1740  break;
1741  }
1742  }
1743 
1744  g_list_free (all);
1745 
1746  return retval;
1747 }
1748 
1749 
1750 /********************************************************************
1751  * gnc_commodity_table_insert
1752  * add a commodity to the table.
1753  ********************************************************************/
1754 
1755 gnc_commodity *
1756 gnc_commodity_table_insert(gnc_commodity_table * table,
1757  gnc_commodity * comm)
1758 {
1759  gnc_commodity_namespace * nsp = nullptr;
1760  gnc_commodity *c;
1761  const char *ns_name;
1762  gnc_commodityPrivate* priv;
1763  QofBook *book;
1764 
1765  if (!table) return nullptr;
1766  if (!comm) return nullptr;
1767 
1768  priv = GET_PRIVATE(comm);
1769 
1770  ENTER ("(table=%p, comm=%p) %s %s", table, comm,
1771  (priv->mnemonic == nullptr ? "(null)" : priv->mnemonic),
1772  (priv->fullname == nullptr ? "(null)" : priv->fullname));
1773  ns_name = gnc_commodity_namespace_get_name(priv->name_space);
1774  c = gnc_commodity_table_lookup (table, ns_name, priv->mnemonic);
1775 
1776  if (c)
1777  {
1778  if (c == comm)
1779  {
1780  LEAVE("already in table");
1781  return c;
1782  }
1783 
1784  /* Backward compatibility support for currencies that have
1785  * recently changed. */
1786  if (priv->name_space->iso4217)
1787  {
1788  auto it = gnc_new_iso_codes.find (priv->mnemonic);
1789  if (it != gnc_new_iso_codes.end())
1790  gnc_commodity_set_mnemonic(comm, it->second.c_str());
1791  }
1792  gnc_commodity_copy (c, comm);
1793  gnc_commodity_destroy (comm);
1794  LEAVE("found at %p", c);
1795  return c;
1796  }
1797 
1798  /* Prevent setting anything except template in namespace template. */
1799  if (g_strcmp0 (ns_name, GNC_COMMODITY_NS_TEMPLATE) == 0 &&
1800  g_strcmp0 (priv->mnemonic, "template") != 0)
1801  {
1802  PWARN("Converting commodity %s from namespace template to "
1803  "namespace User", priv->mnemonic);
1804  gnc_commodity_set_namespace (comm, "User");
1805  ns_name = "User";
1806  mark_commodity_dirty (comm);
1807  }
1808 
1809  book = qof_instance_get_book (&comm->inst);
1810  nsp = gnc_commodity_table_add_namespace(table, ns_name, book);
1811 
1812  PINFO ("insert %p %s into nsp=%p %s", priv->mnemonic, priv->mnemonic,
1813  nsp->cm_table, nsp->name);
1814  g_hash_table_insert(nsp->cm_table,
1815  (gpointer)CACHE_INSERT(priv->mnemonic),
1816  (gpointer)comm);
1817  nsp->cm_list = g_list_append(nsp->cm_list, comm);
1818 
1819  qof_event_gen (&comm->inst, QOF_EVENT_ADD, nullptr);
1820  LEAVE ("(table=%p, comm=%p)", table, comm);
1821  return comm;
1822 }
1823 
1824 /********************************************************************
1825  * gnc_commodity_table_remove
1826  * remove a commodity from the table.
1827  ********************************************************************/
1828 
1829 void
1830 gnc_commodity_table_remove(gnc_commodity_table * table,
1831  gnc_commodity * comm)
1832 {
1833  gnc_commodity_namespace * nsp;
1834  gnc_commodity *c;
1835  gnc_commodityPrivate* priv;
1836  const char *ns_name;
1837 
1838  if (!table) return;
1839  if (!comm) return;
1840 
1841  priv = GET_PRIVATE(comm);
1842  ns_name = gnc_commodity_namespace_get_name(priv->name_space);
1843  c = gnc_commodity_table_lookup (table, ns_name, priv->mnemonic);
1844  if (c != comm) return;
1845 
1846  qof_event_gen (&comm->inst, QOF_EVENT_REMOVE, nullptr);
1847 
1848  nsp = gnc_commodity_table_find_namespace(table, ns_name);
1849  if (!nsp) return;
1850 
1851  nsp->cm_list = g_list_remove(nsp->cm_list, comm);
1852  g_hash_table_remove (nsp->cm_table, priv->mnemonic);
1853  /* XXX minor mem leak, should remove the key as well */
1854 }
1855 
1856 /********************************************************************
1857  * gnc_commodity_table_has_namespace
1858  * see if the commodities namespace exists. May have zero commodities.
1859  ********************************************************************/
1860 
1861 int
1862 gnc_commodity_table_has_namespace(const gnc_commodity_table * table,
1863  const char * name_space)
1864 {
1865  gnc_commodity_namespace * nsp = nullptr;
1866 
1867  if (!table || !name_space)
1868  {
1869  return 0;
1870  }
1871 
1872  nsp = gnc_commodity_table_find_namespace(table, name_space);
1873  if (nsp)
1874  {
1875  return 1;
1876  }
1877  else
1878  {
1879  return 0;
1880  }
1881 }
1882 
1883 static void
1884 hash_keys_helper(gpointer key, gpointer value, gpointer data)
1885 {
1886  auto l = (GList**)data;
1887  *l = g_list_prepend(*l, key);
1888 }
1889 
1890 static GList *
1891 g_hash_table_keys(GHashTable * table)
1892 {
1893  GList * l = nullptr;
1894  g_hash_table_foreach(table, &hash_keys_helper, (gpointer) &l);
1895  return l;
1896 }
1897 
1898 static void
1899 hash_values_helper(gpointer key, gpointer value, gpointer data)
1900 {
1901  auto l = (GList**)data;
1902  *l = g_list_prepend(*l, value);
1903 }
1904 
1905 static GList *
1906 g_hash_table_values(GHashTable * table)
1907 {
1908  GList * l = nullptr;
1909  g_hash_table_foreach(table, &hash_values_helper, (gpointer) &l);
1910  return l;
1911 }
1912 
1913 /********************************************************************
1914  * gnc_commodity_table_get_namespaces
1915  * see if any commodities in the namespace exist
1916  ********************************************************************/
1917 
1918 GList *
1919 gnc_commodity_table_get_namespaces(const gnc_commodity_table * table)
1920 {
1921  if (!table)
1922  return nullptr;
1923 
1924  return g_hash_table_keys(table->ns_table);
1925 }
1926 
1927 GList *
1929 {
1930  if (!table)
1931  return nullptr;
1932 
1933  return g_list_copy (table->ns_list);
1934 }
1935 
1936 /* Because gnc_commodity_table_add_namespace maps GNC_COMMODITY_NS_ISO to
1937  GNC_COMMODITY_NS_CURRENCY and then sets iso4217 if the namespace is
1938  either of these, the net result is that the iso4217 bit is set only
1939  for GNC_COMMODITY_NS_CURRENCY. This means that gnc_commodity_is_iso is
1940  a subset of gnc_commodity_is_currency. Most callers seem to use
1941  gnc_commodity_is_iso. */
1942 gboolean
1943 gnc_commodity_is_iso(const gnc_commodity * cm)
1944 {
1945  gnc_commodityPrivate* priv;
1946 
1947  if (!cm) return FALSE;
1948 
1949  priv = GET_PRIVATE(cm);
1950  if ( !priv->name_space) return FALSE;
1951  return priv->name_space->iso4217;
1952 }
1953 
1954 gboolean
1955 gnc_commodity_is_currency(const gnc_commodity *cm)
1956 {
1957  const char *ns_name;
1958  if (!cm) return FALSE;
1959 
1960  ns_name = gnc_commodity_namespace_get_name(GET_PRIVATE(cm)->name_space);
1961  return (!g_strcmp0(ns_name, GNC_COMMODITY_NS_LEGACY) ||
1962  !g_strcmp0(ns_name, GNC_COMMODITY_NS_CURRENCY));
1963 }
1964 
1965 /********************************************************************
1966  * gnc_commodity_table_get_commodities
1967  * list commodities in a given namespace
1968  ********************************************************************/
1969 
1970 static CommodityList*
1971 commodity_table_get_all_noncurrency_commodities(const gnc_commodity_table* table)
1972 {
1973  GList *node = nullptr, *nslist = gnc_commodity_table_get_namespaces(table);
1974  CommodityList *retval = nullptr;
1975  for (node = nslist; node; node=g_list_next(node))
1976  {
1977  gnc_commodity_namespace *ns = nullptr;
1978  if (g_strcmp0((char*)(node->data), GNC_COMMODITY_NS_CURRENCY) == 0
1979  || g_strcmp0((char*)(node->data), GNC_COMMODITY_NS_TEMPLATE) == 0)
1980  continue;
1981  ns = gnc_commodity_table_find_namespace(table, (char*)(node->data));
1982  if (!ns)
1983  continue;
1984  retval = g_list_concat(g_hash_table_values(ns->cm_table), retval);
1985  }
1986  g_list_free(nslist);
1987  return retval;
1988 }
1989 
1990 CommodityList *
1991 gnc_commodity_table_get_commodities(const gnc_commodity_table * table,
1992  const char * name_space)
1993 {
1994  gnc_commodity_namespace * ns = nullptr;
1995 
1996  if (!table)
1997  return nullptr;
1998  if (g_strcmp0(name_space, GNC_COMMODITY_NS_NONISO_GUI) == 0)
1999  return commodity_table_get_all_noncurrency_commodities(table);
2000  ns = gnc_commodity_table_find_namespace(table, name_space);
2001  if (!ns)
2002  return nullptr;
2003 
2004  return g_hash_table_values(ns->cm_table);
2005 }
2006 
2007 /********************************************************************
2008  * gnc_commodity_table_get_quotable_commodities
2009  * list commodities in a given namespace that get price quotes
2010  ********************************************************************/
2011 
2012 static void
2013 get_quotables_helper1(gpointer key, gpointer value, gpointer data)
2014 {
2015  auto comm = GNC_COMMODITY(value);
2016  gnc_commodityPrivate* priv = GET_PRIVATE(comm);
2017  auto l = static_cast<GList**>(data);
2018 
2019  if (!priv->quote_flag || !priv->quote_source || !priv->quote_source->get_supported())
2020  return;
2021  *l = g_list_prepend(*l, value);
2022 }
2023 
2024 static gboolean
2025 get_quotables_helper2 (gnc_commodity *comm, gpointer data)
2026 {
2027  auto l = static_cast<GList**>(data);
2028  gnc_commodityPrivate* priv = GET_PRIVATE(comm);
2029 
2030  if (!priv->quote_flag || priv->quote_source || !priv->quote_source->get_supported())
2031  return TRUE;
2032  *l = g_list_prepend(*l, comm);
2033  return TRUE;
2034 }
2035 
2036 CommodityList *
2038 {
2039  gnc_commodity_namespace * ns = nullptr;
2040  const char *name_space;
2041  GList * nslist, * tmp;
2042  GList * l = nullptr;
2043  regex_t pattern;
2044  const char *expression = gnc_prefs_get_namespace_regexp();
2045 
2046  ENTER("table=%p, expression=%s", table, expression);
2047  if (!table)
2048  return nullptr;
2049 
2050  if (expression && *expression)
2051  {
2052  if (regcomp(&pattern, expression, REG_EXTENDED | REG_ICASE) != 0)
2053  {
2054  LEAVE("Cannot compile regex");
2055  return nullptr;
2056  }
2057 
2059  for (tmp = nslist; tmp; tmp = tmp->next)
2060  {
2061  name_space = static_cast<const char*>(tmp->data);
2062  if (regexec(&pattern, name_space, 0, nullptr, 0) == 0)
2063  {
2064  DEBUG("Running list of %s commodities", name_space);
2065  ns = gnc_commodity_table_find_namespace(table, name_space);
2066  if (ns)
2067  {
2068  g_hash_table_foreach(ns->cm_table, &get_quotables_helper1, (gpointer) &l);
2069  }
2070  }
2071  }
2072  g_list_free(nslist);
2073  regfree(&pattern);
2074  }
2075  else
2076  {
2077  gnc_commodity_table_foreach_commodity(table, get_quotables_helper2,
2078  (gpointer) &l);
2079  }
2080  LEAVE("list head %p", l);
2081  return l;
2082 }
2083 
2084 /********************************************************************
2085  * gnc_commodity_table_add_namespace
2086  * add an empty namespace if it does not exist
2087  ********************************************************************/
2088 
2089 /* GObject Initialization */
2090 QOF_GOBJECT_IMPL(gnc_commodity_namespace, gnc_commodity_namespace, QOF_TYPE_INSTANCE)
2091 
2092 static void
2093 gnc_commodity_namespace_init(gnc_commodity_namespace* ns)
2094 {
2095 }
2096 
2097 static void
2098 gnc_commodity_namespace_dispose_real (GObject *nsp)
2099 {
2100 }
2101 
2102 static void
2103 gnc_commodity_namespace_finalize_real(GObject* nsp)
2104 {
2105 }
2106 
2107 gnc_commodity_namespace *
2109  const char * name_space,
2110  QofBook *book)
2111 {
2112  gnc_commodity_namespace * ns = nullptr;
2113 
2114  if (!table) return nullptr;
2115 
2116  name_space = gnc_commodity_table_map_namespace(name_space);
2117  ns = gnc_commodity_table_find_namespace(table, name_space);
2118  if (!ns)
2119  {
2120  ns = static_cast<gnc_commodity_namespace*>(g_object_new(GNC_TYPE_COMMODITY_NAMESPACE, nullptr));
2121  ns->cm_table = g_hash_table_new(g_str_hash, g_str_equal);
2122  ns->name = CACHE_INSERT(static_cast<const char*>(name_space));
2123  ns->iso4217 = gnc_commodity_namespace_is_iso(name_space);
2124  qof_instance_init_data (&ns->inst, GNC_ID_COMMODITY_NAMESPACE, book);
2125  qof_event_gen (&ns->inst, QOF_EVENT_CREATE, nullptr);
2126 
2127  g_hash_table_insert(table->ns_table,
2128  (gpointer) ns->name,
2129  (gpointer) ns);
2130  table->ns_list = g_list_append(table->ns_list, ns);
2131  qof_event_gen (&ns->inst, QOF_EVENT_ADD, nullptr);
2132  }
2133  return ns;
2134 }
2135 
2136 
2137 gnc_commodity_namespace *
2138 gnc_commodity_table_find_namespace(const gnc_commodity_table * table,
2139  const char * name_space)
2140 {
2141  if (!table || !name_space)
2142  return nullptr;
2143 
2144  name_space = gnc_commodity_table_map_namespace(name_space);
2145  return static_cast<gnc_commodity_namespace*>(g_hash_table_lookup(table->ns_table, (gpointer)name_space));
2146 }
2147 
2148 
2149 gnc_commodity *
2150 gnc_commodity_find_commodity_by_guid(const GncGUID *guid, QofBook *book)
2151 {
2152  QofCollection *col;
2153  if (!guid || !book) return nullptr;
2154  col = qof_book_get_collection (book, GNC_ID_COMMODITY);
2155  return (gnc_commodity *) qof_collection_lookup_entity (col, guid);
2156 }
2157 
2158 /********************************************************************
2159  * gnc_commodity_table_delete_namespace
2160  * delete a namespace
2161  ********************************************************************/
2162 
2163 static int
2164 ns_helper(gpointer key, gpointer value, gpointer user_data)
2165 {
2166  auto c = GNC_COMMODITY(value);
2168  CACHE_REMOVE(static_cast<char*>(key)); /* key is commodity mnemonic */
2169  return TRUE;
2170 }
2171 
2172 void
2174  const char * name_space)
2175 {
2176  gnc_commodity_namespace * ns;
2177 
2178  if (!table) return;
2179 
2180  ns = gnc_commodity_table_find_namespace(table, name_space);
2181  if (!ns)
2182  return;
2183 
2184  qof_event_gen (&ns->inst, QOF_EVENT_REMOVE, nullptr);
2185  g_hash_table_remove(table->ns_table, name_space);
2186  table->ns_list = g_list_remove(table->ns_list, ns);
2187 
2188  g_list_free(ns->cm_list);
2189  ns->cm_list = nullptr;
2190 
2191  g_hash_table_foreach_remove(ns->cm_table, ns_helper, nullptr);
2192  g_hash_table_destroy(ns->cm_table);
2193  CACHE_REMOVE(ns->name);
2194 
2195  qof_event_gen (&ns->inst, QOF_EVENT_DESTROY, nullptr);
2196  /* qof_instance_release(&ns->inst); */
2197  g_object_unref(ns);
2198 }
2199 
2200 /********************************************************************
2201  * gnc_commodity_table_foreach_commodity
2202  * call user-defined function once for every commodity in every
2203  * namespace
2204  ********************************************************************/
2205 
2206 typedef struct
2207 {
2208  gboolean ok;
2209  gboolean (*func)(gnc_commodity *, gpointer);
2210  gpointer user_data;
2211 } IterData;
2212 
2213 static void
2214 iter_commodity (gpointer key, gpointer value, gpointer user_data)
2215 {
2216  IterData *iter_data = (IterData *) user_data;
2217  gnc_commodity *cm = (gnc_commodity *) value;
2218 
2219  if (iter_data->ok)
2220  {
2221  iter_data->ok = (iter_data->func)(cm, iter_data->user_data);
2222  }
2223 }
2224 
2225 static void
2226 iter_namespace (gpointer key, gpointer value, gpointer user_data)
2227 {
2228  GHashTable *namespace_hash = ((gnc_commodity_namespace *) value)->cm_table;
2229  g_hash_table_foreach (namespace_hash, iter_commodity, user_data);
2230 }
2231 
2232 gboolean
2233 gnc_commodity_table_foreach_commodity (const gnc_commodity_table * tbl,
2234  gboolean (*f)(gnc_commodity *, gpointer),
2235  gpointer user_data)
2236 {
2237  IterData iter_data;
2238 
2239  if (!tbl || !f) return FALSE;
2240 
2241  iter_data.ok = TRUE;
2242  iter_data.func = f;
2243  iter_data.user_data = user_data;
2244 
2245  g_hash_table_foreach(tbl->ns_table, iter_namespace, (gpointer)&iter_data);
2246 
2247  return iter_data.ok;
2248 }
2249 
2250 /********************************************************************
2251  * gnc_commodity_table_destroy
2252  * cleanup and free.
2253  ********************************************************************/
2254 
2255 void
2256 gnc_commodity_table_destroy(gnc_commodity_table * t)
2257 {
2258  gnc_commodity_namespace * ns;
2259  GList *item, *next;
2260 
2261  if (!t) return;
2262  ENTER ("table=%p", t);
2263 
2264  for (item = t->ns_list; item; item = next)
2265  {
2266  next = g_list_next(item);
2267  ns = static_cast<gnc_commodity_namespace*>(item->data);
2269  }
2270 
2271  g_list_free(t->ns_list);
2272  t->ns_list = nullptr;
2273  g_hash_table_destroy(t->ns_table);
2274  t->ns_table = nullptr;
2275  LEAVE ("table=%p", t);
2276  g_free(t);
2277 }
2278 
2279 /* =========================================================== */
2280 
2281 /********************************************************************
2282  * gnc_commodity_table_add_default_data
2283  ********************************************************************/
2284 
2285 #define CUR_I18N(String) dgettext ("iso_4217", String)
2286 
2287 gboolean
2288 gnc_commodity_table_add_default_data(gnc_commodity_table *table, QofBook *book)
2289 {
2290  QofCollection *col;
2291  gnc_commodity* c;
2292 
2293  ENTER ("table=%p", table);
2294  gnc_commodity_table_add_namespace(table, GNC_COMMODITY_NS_TEMPLATE, book);
2295  c = gnc_commodity_new(book, "template", GNC_COMMODITY_NS_TEMPLATE, "template", "template", 1);
2297 
2298 #include "iso-4217-currencies.c"
2299 
2300  /* We've just created the default namespaces and currencies. Mark
2301  * these collections as clean because there is no USER entered data
2302  * in these collections as of yet. */
2303  col = qof_book_get_collection(book, GNC_ID_COMMODITY);
2305  col = qof_book_get_collection(book, GNC_ID_COMMODITY_NAMESPACE);
2307 
2308  LEAVE ("table=%p", table);
2309  return TRUE;
2310 }
2311 
2312 /********************************************************************
2313  ********************************************************************/
2314 /* QofObject function implementation and registration */
2315 
2316 #ifdef _MSC_VER
2317 /* MSVC compiler doesn't have C99 "designated initializers"
2318  * so we wrap them in a macro that is empty on MSVC. */
2319 # define DI(x) /* */
2320 #else
2321 # define DI(x) x
2322 #endif
2323 static QofObject commodity_object_def =
2324 {
2325  DI(.interface_version = ) QOF_OBJECT_VERSION,
2326  DI(.e_type = ) GNC_ID_COMMODITY,
2327  DI(.type_label = ) "Commodity",
2328  DI(.create = ) nullptr,
2329  DI(.book_begin = ) nullptr,
2330  DI(.book_end = ) nullptr,
2331  DI(.is_dirty = ) qof_collection_is_dirty,
2332  DI(.mark_clean = ) qof_collection_mark_clean,
2333  DI(.foreach = ) qof_collection_foreach,
2334  DI(.printable = ) (const char * (*)(gpointer)) gnc_commodity_get_fullname,
2335 };
2336 
2337 static QofObject namespace_object_def =
2338 {
2339  DI(.interface_version = ) QOF_OBJECT_VERSION,
2340  DI(.e_type = ) GNC_ID_COMMODITY_NAMESPACE,
2341  DI(.type_label = ) "Namespace",
2342  DI(.create = ) nullptr,
2343  DI(.book_begin = ) nullptr,
2344  DI(.book_end = ) nullptr,
2345  DI(.is_dirty = ) nullptr,
2346  DI(.mark_clean = ) nullptr,
2347  DI(.foreach = ) nullptr,
2348  DI(.printable = ) nullptr,
2349 };
2350 
2351 static void
2352 commodity_table_book_begin (QofBook *book)
2353 {
2354  gnc_commodity_table *ct;
2355  ENTER ("book=%p", book);
2356 
2358  return;
2359 
2360  ct = gnc_commodity_table_new ();
2361  qof_book_set_data (book, GNC_COMMODITY_TABLE, ct);
2362 
2363  if (!gnc_commodity_table_add_default_data(ct, book))
2364  {
2365  PWARN("unable to initialize book's commodity_table");
2366  }
2367 
2368  LEAVE ("book=%p", book);
2369 }
2370 
2371 static void
2372 commodity_table_book_end (QofBook *book)
2373 {
2374  gnc_commodity_table *ct;
2375 
2376  ct = gnc_commodity_table_get_table (book);
2377  qof_book_set_data (book, GNC_COMMODITY_TABLE, nullptr);
2378  gnc_commodity_table_destroy (ct);
2379 }
2380 
2381 static QofObject commodity_table_object_def =
2382 {
2383  DI(.interface_version = ) QOF_OBJECT_VERSION,
2384  DI(.e_type = ) GNC_ID_COMMODITY_TABLE,
2385  DI(.type_label = ) "CommodityTable",
2386  DI(.create = ) nullptr,
2387  DI(.book_begin = ) commodity_table_book_begin,
2388  DI(.book_end = ) commodity_table_book_end,
2389  DI(.is_dirty = ) qof_collection_is_dirty,
2390  DI(.mark_clean = ) qof_collection_mark_clean,
2391  DI(.foreach = ) nullptr,
2392  DI(.printable = ) nullptr,
2393  DI(.version_cmp = ) nullptr,
2394 };
2395 
2396 gboolean
2398 {
2399  if (!qof_object_register (&commodity_object_def))
2400  return FALSE;
2401  if (!qof_object_register (&namespace_object_def))
2402  return FALSE;
2403  return qof_object_register (&commodity_table_object_def);
2404 }
2405 
2406 /* *******************************************************************
2407 * gnc_monetary methods
2408 ********************************************************************/
2409 
2411 MonetaryList *
2412 gnc_monetary_list_add_monetary(MonetaryList *list, gnc_monetary add_mon)
2413 {
2414  MonetaryList *l = list, *tmp;
2415  for (tmp = list; tmp; tmp = tmp->next)
2416  {
2417  auto list_mon = static_cast<gnc_monetary*>(tmp->data);
2418  if (gnc_commodity_equiv(list_mon->commodity, add_mon.commodity))
2419  {
2420  list_mon->value = gnc_numeric_add(list_mon->value, add_mon.value,
2422  break;
2423  }
2424  }
2425 
2426  /* See if we found an entry, and add one if not */
2427  if (tmp == nullptr)
2428  {
2429  auto new_mon = static_cast<gnc_monetary*>(g_new0(gnc_monetary, 1));
2430  *new_mon = add_mon;
2431  l = g_list_prepend(l, new_mon);
2432  }
2433 
2434  return l;
2435 }
2436 
2439 MonetaryList *
2441 {
2442  MonetaryList *node, *next;
2443  for (node = list; node; node = next)
2444  {
2445  auto mon = static_cast<gnc_monetary*>(node->data);
2446  next = node->next;
2447  if (gnc_numeric_zero_p(mon->value))
2448  {
2449  g_free(mon);
2450  list = g_list_delete_link(list, node);
2451  }
2452  }
2453  return list;
2454 }
2455 
2457 void
2458 gnc_monetary_list_free(MonetaryList *list)
2459 {
2460  MonetaryList *tmp;
2461  for (tmp = list; tmp; tmp = tmp->next)
2462  {
2463  g_free(tmp->data);
2464  }
2465 
2466  g_list_free(list);
2467 }
2468 
2469 /* ========================= END OF FILE ============================== */
gnc_commodity * gnc_commodity_table_insert(gnc_commodity_table *table, gnc_commodity *comm)
Add a new commodity to the commodity table.
const char * gnc_commodity_get_cusip(const gnc_commodity *cm)
Retrieve the &#39;exchange code&#39; for the specified commodity.
gboolean gnc_commodity_table_foreach_commodity(const gnc_commodity_table *table, gboolean(*f)(gnc_commodity *cm, gpointer user_data), gpointer user_data)
Call a function once for each commodity in the commodity table.
gnc_commodity_table * gnc_commodity_table_get_table(QofBook *book)
Returns the commodity table associated with a book.
A gnc_commodity_table is a database of commodity info.
gboolean gnc_commodity_is_currency(const gnc_commodity *cm)
Checks to see if the specified commodity is an ISO 4217 recognized currency or a legacy currency...
int gnc_commodity_get_fraction(const gnc_commodity *cm)
Retrieve the fraction for the specified commodity.
gboolean gnc_commodity_table_add_default_data(gnc_commodity_table *table, QofBook *book)
Add all the standard namespaces and currencies to the commodity table.
const char * gnc_quote_source_get_user_name(const gnc_quote_source *source)
Given a gnc_quote_source data structure, return the user friendly name of this quote source...
const char * gnc_commodity_get_mnemonic(const gnc_commodity *cm)
Retrieve the mnemonic for the specified commodity.
#define GNC_COMMODITY_MAX_FRACTION
Max fraction is 10^9 because 10^10 would require changing it to an int64_t.
This quote source pulls from a single specific web site.
QofBook * qof_instance_get_book(gconstpointer inst)
Return the book pointer.
gboolean qof_collection_is_dirty(const QofCollection *col)
Return value of &#39;dirty&#39; flag on collection.
Definition: qofid.cpp:255
gnc_quote_source * gnc_quote_source_add_new(const char *source_name, gboolean supported)
Create a new quote source.
const char * gnc_commodity_namespace_get_gui_name(const gnc_commodity_namespace *ns)
Return the textual name of a namespace data structure in a form suitable to present to the user...
QofInstance * qof_collection_lookup_entity(const QofCollection *col, const GncGUID *guid)
Find the entity going only from its guid.
Definition: qofid.cpp:212
#define PINFO(format, args...)
Print an informational note.
Definition: qoflog.h:256
gboolean gnc_commodity_get_quote_flag(const gnc_commodity *cm)
Retrieve the automatic price quote flag for the specified commodity.
QofBackendError
The errors that can be reported to the GUI & other front-end users.
Definition: qofbackend.h:57
const char * gnc_commodity_get_user_symbol(const gnc_commodity *cm)
Retrieve the user-defined symbol for the specified commodity.
void gnc_commodity_set_quote_tz(gnc_commodity *cm, const char *tz)
Set the automatic price quote timezone for the specified commodity.
void gnc_commodity_decrement_usage_count(gnc_commodity *cm)
Decrement a commodity&#39;s internal counter that tracks how many accounts are using that commodity...
Commodity handling public routines (C++ api)
const char * gnc_commodity_get_quote_tz(const gnc_commodity *cm)
Retrieve the automatic price quote timezone for the specified commodity.
#define DEBUG(format, args...)
Print a debugging message.
Definition: qoflog.h:264
void gnc_commodity_set_fraction(gnc_commodity *cm, int fraction)
Set the fraction for the specified commodity.
gboolean gnc_commodity_equal(const gnc_commodity *a, const gnc_commodity *b)
This routine returns TRUE if the two commodities are equal.
gboolean gnc_quote_source_get_supported(const gnc_quote_source *source)
Given a gnc_quote_source data structure, return the flag that indicates whether this particular quote...
globally unique ID User API
gnc_numeric gnc_numeric_add(gnc_numeric a, gnc_numeric b, gint64 denom, gint how)
Return a+b.
QuoteSourceType gnc_quote_source_get_type(const gnc_quote_source *source)
Given a gnc_quote_source data structure, return the type of this particular quote source...
The special currency quote source.
int gnc_commodity_compare_void(const void *a, const void *b)
A wrapper around gnc_commodity_compare() which offers the function declaration that is needed for g_l...
gboolean gnc_numeric_zero_p(gnc_numeric a)
Returns 1 if the given gnc_numeric is 0 (zero), else returns 0.
Object instance holds common fields that most gnucash objects use.
gnc_commodity * gnc_commodity_clone(const gnc_commodity *src, QofBook *dest_book)
allocate and copy
This is a locally installed quote source that gnucash knows nothing about.
const char * gnc_commodity_get_namespace(const gnc_commodity *cm)
Retrieve the namespace for the specified commodity.
Use any denominator which gives an exactly correct ratio of numerator to denominator.
Definition: gnc-numeric.h:188
A gnc_commodity_namespace is an collection of commodities.
void gnc_commodity_set_quote_flag(gnc_commodity *cm, const gboolean flag)
Set the automatic price quote flag for the specified commodity.
#define QOF_OBJECT_VERSION
Defines the version of the core object object registration interface.
Definition: qofobject.h:63
QuoteSourceType
The quote source type enum account types are used to determine how the transaction data in the accoun...
gnc_quote_source * gnc_quote_source_lookup_by_ti(QuoteSourceType type, gint index)
Given the type/index of a quote source, find the data structure identified by this pair...
gboolean qof_commit_edit(QofInstance *inst)
commit_edit helpers
#define PERR(format, args...)
Log a serious error.
Definition: qoflog.h:244
#define ENTER(format, args...)
Print a function entry debugging message.
Definition: qoflog.h:272
void gnc_commodity_user_set_quote_flag(gnc_commodity *cm, const gboolean flag)
Set the automatic price quote flag for the specified commodity, based on user input.
gnc_commodity_namespace * gnc_commodity_table_add_namespace(gnc_commodity_table *table, const char *name_space, QofBook *book)
This function adds a new string to the list of commodity namespaces.
GList * gnc_commodity_namespace_get_commodity_list(const gnc_commodity_namespace *name_space)
Return a list of all commodity data structures in the specified namespace.
void gnc_commodity_set_user_symbol(gnc_commodity *cm, const char *user_symbol)
Set a user-defined symbol for the specified commodity.
const char * gnc_commodity_namespace_get_name(const gnc_commodity_namespace *ns)
Return the textual name of a namespace data structure.
#define PWARN(format, args...)
Log a warning.
Definition: qoflog.h:250
void qof_instance_init_data(QofInstance *inst, QofIdType type, QofBook *book)
Initialise the settings associated with an instance.
MonetaryList * gnc_monetary_list_delete_zeros(MonetaryList *list)
Delete all entries in the list that have zero value.
gboolean qof_begin_edit(QofInstance *inst)
begin_edit
void gnc_commodity_set_quote_source(gnc_commodity *cm, gnc_quote_source *src)
Set the automatic price quote source for the specified commodity.
gint gnc_quote_source_num_entries(QuoteSourceType type)
Return the number of entries for a given type of quote source.
GList * gnc_commodity_table_get_namespaces(const gnc_commodity_table *table)
Return a list of all namespaces in the commodity table.
gboolean gnc_commodity_table_register(void)
You should probably not be using gnc_commodity_table_register() It is an internal routine for registe...
void gnc_commodity_increment_usage_count(gnc_commodity *cm)
Increment a commodity&#39;s internal counter that tracks how many accounts are using that commodity...
void gnc_commodity_set_cusip(gnc_commodity *cm, const char *cusip)
Set the &#39;exchange code&#39; for the specified commodity.
void gnc_monetary_list_free(MonetaryList *list)
Free a MonetaryList and all the monetaries it points to.
gnc_commodity * gnc_commodity_new(QofBook *book, const char *fullname, const char *name_space, const char *mnemonic, const char *cusip, int fraction)
Create a new commodity.
int gnc_commodity_table_has_namespace(const gnc_commodity_table *table, const char *name_space)
Test to see if the indicated namespace exits in the commodity table.
void gnc_commodity_table_delete_namespace(gnc_commodity_table *table, const char *name_space)
This function deletes a string from the list of commodity namespaces.
gboolean gnc_commodity_namespace_is_iso(const char *name_space)
Checks to see if the specified commodity namespace is the namespace for ISO 4217 currencies.
const char * gnc_commodity_get_fullname(const gnc_commodity *cm)
Retrieve the full name for the specified commodity.
#define GNC_COMMODITY_NS_LEGACY
The commodity namespace definitions are used to tag a commodity by its type, or a stocks by the excha...
void qof_book_set_data(QofBook *book, const gchar *key, gpointer data)
The qof_book_set_data() allows arbitrary pointers to structs to be stored in QofBook.
gboolean qof_commit_edit_part2(QofInstance *inst, void(*on_error)(QofInstance *, QofBackendError), void(*on_done)(QofInstance *), void(*on_free)(QofInstance *))
part2 – deal with the backend
const char * gnc_commodity_get_nice_symbol(const gnc_commodity *cm)
Retrieve a symbol for the specified commodity, suitable for display to the user.
void qof_collection_mark_clean(QofCollection *)
reset value of dirty flag
Definition: qofid.cpp:261
void gnc_quote_source_set_fq_installed(const char *version_string, const std::vector< std::string > &sources_list)
Update gnucash internal tables based on what Finance::Quote sources are installed.
const char * gnc_quote_source_fq_version(void)
This function returns the version of the Finance::Quote module installed on a user&#39;s computer...
Generic api to store and retrieve preferences.
CommodityList * gnc_commodity_table_get_commodities(const gnc_commodity_table *table, const char *name_space)
Return a list of all commodities in the commodity table that are in the given namespace.
gnc_quote_source * gnc_quote_source_lookup_by_internal(const char *name)
Given the internal (gnucash or F::Q) name of a quote source, find the data structure identified by th...
const char * gnc_commodity_get_printname(const gnc_commodity *cm)
Retrieve the &#39;print&#39; name for the specified commodity.
int gnc_commodity_compare(const gnc_commodity *a, const gnc_commodity *b)
This routine returns 0 if the two commodities are equal, 1 otherwise.
void gnc_commodity_set_fullname(gnc_commodity *cm, const char *fullname)
Set the full name for the specified commodity.
This quote source may pull from multiple web sites.
gnc_quote_source * gnc_commodity_get_quote_source(const gnc_commodity *cm)
Retrieve the automatic price quote source for the specified commodity.
gnc_commodity_namespace * gnc_commodity_table_find_namespace(const gnc_commodity_table *table, const char *name_space)
This function finds a commodity namespace in the set of existing commodity namespaces.
void gnc_commodity_set_mnemonic(gnc_commodity *cm, const char *mnemonic)
Set the mnemonic for the specified commodity.
const char * gnc_commodity_get_default_symbol(const gnc_commodity *cm)
Retrieve the default symbol for the specified commodity.
gint qof_instance_guid_compare(gconstpointer ptr1, gconstpointer ptr2)
Compare the GncGUID values of two instances.
CommodityList * gnc_commodity_table_get_quotable_commodities(const gnc_commodity_table *table)
This function returns a list of commodities for which price quotes should be retrieved.
gnc_commodity_table * gnc_commodity_table_new(void)
You probably shouldn&#39;t be using gnc_commodity_table_new() directly, it&#39;s for internal use only...
gint gnc_quote_source_get_index(const gnc_quote_source *source)
Given a gnc_quote_source data structure, return the index of this particular quote source within its ...
gnc_commodity_namespace * gnc_commodity_get_namespace_ds(const gnc_commodity *cm)
Retrieve the namespace data structure for the specified commodity.
#define LEAVE(format, args...)
Print a function exit debugging message.
Definition: qoflog.h:282
const char * gnc_commodity_get_unique_name(const gnc_commodity *cm)
Retrieve the &#39;unique&#39; name for the specified commodity.
MonetaryList * gnc_monetary_list_add_monetary(MonetaryList *list, gnc_monetary add_mon)
Add a gnc_monetary to the list.
void gnc_commodity_table_remove(gnc_commodity_table *table, gnc_commodity *comm)
Remove a commodity from the commodity table.
void gnc_commodity_set_namespace(gnc_commodity *cm, const char *name_space)
Set the namespace for the specified commodity.
QofCollection * qof_book_get_collection(const QofBook *book, QofIdType entity_type)
Return The table of entities of the given type.
Definition: qofbook.cpp:521
guint gnc_commodity_table_get_size(const gnc_commodity_table *tbl)
Returns the number of commodities in the commodity table.
const char * gnc_quote_source_get_internal_name(const gnc_quote_source *source)
Given a gnc_quote_source data structure, return the internal name of this quote source.
gboolean qof_object_register(const QofObject *object)
Register new types of object objects.
Definition: qofobject.cpp:299
An article that is bought and sold.
void qof_event_gen(QofInstance *entity, QofEventId event_id, gpointer event_data)
Invoke all registered event handlers using the given arguments.
Definition: qofevent.cpp:231
#define GNC_DENOM_AUTO
Values that can be passed as the &#39;denom&#39; argument.
Definition: gnc-numeric.h:245
The type used to store guids in C.
Definition: guid.h:75
void gnc_commodity_copy(gnc_commodity *dest, const gnc_commodity *src)
Copy src into dest.
gpointer qof_book_get_data(const QofBook *book, const gchar *key)
Retrieves arbitrary pointers to structs stored by qof_book_set_data.
GList * gnc_commodity_table_get_namespaces_list(const gnc_commodity_table *table)
Return a list of all namespace data structures in the commodity table.
gnc_commodity * gnc_commodity_obtain_twin(const gnc_commodity *from, QofBook *book)
Given the commodity &#39;findlike&#39;, this routine will find and return the equivalent commodity (commodity...
Commodity handling public routines.
void gnc_commodity_destroy(gnc_commodity *cm)
Destroy a commodity.
gboolean gnc_commodity_equiv(const gnc_commodity *a, const gnc_commodity *b)
This routine returns TRUE if the two commodities are equivalent.
gboolean gnc_commodity_is_iso(const gnc_commodity *cm)
Checks to see if the specified commodity is an ISO 4217 recognized currency.
gboolean gnc_quote_source_fq_installed(void)
This function indicates whether or not the Finance::Quote module is installed on a user&#39;s computer...