2 This file is part of GNUnet.
3 (C) 2009-2013 Christian Grothoff (and other contributing authors)
5 GNUnet is free software; you can redistribute it and/or modify
6 it under the terms of the GNU General Public License as published
7 by the Free Software Foundation; either version 3, or (at your
8 option) any later version.
10 GNUnet is distributed in the hope that it will be useful, but
11 WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 General Public License for more details.
15 You should have received a copy of the GNU General Public License
16 along with GNUnet; see the file COPYING. If not, write to the
17 Free Software Foundation, Inc., 59 Temple Place - Suite 330,
18 Boston, MA 02111-1307, USA.
22 * @file gnsrecord/gnsrecord_misc.c
23 * @brief MISC functions related to GNS records
24 * @author Martin Schanzenbach
25 * @author Matthias Wachs
26 * @author Christian Grothoff
29 #include "gnunet_util_lib.h"
30 #include "gnunet_constants.h"
31 #include "gnunet_signatures.h"
32 #include "gnunet_arm_service.h"
33 #include "gnunet_gnsrecord_lib.h"
34 #include "gnunet_dnsparser_lib.h"
35 #include "gnunet_tun_lib.h"
38 #define LOG(kind,...) GNUNET_log_from (kind, "gnsrecord",__VA_ARGS__)
41 * Convert a UTF-8 string to UTF-8 lowercase
42 * @param src source string
43 * @return converted result
46 GNUNET_NAMESTORE_normalize_string (const char *src)
48 GNUNET_assert (NULL != src);
49 char *res = strdup (src);
51 GNUNET_STRINGS_utf8_tolower(src, &res);
57 * Convert a zone key to a string (for printing debug messages).
58 * This is one of the very few calls in the entire API that is
61 * @param z the zone key
62 * @return string form; will be overwritten by next call to #GNUNET_NAMESTORE_z2s
65 GNUNET_NAMESTORE_z2s (const struct GNUNET_CRYPTO_EcdsaPublicKey *z)
67 static char buf[sizeof (struct GNUNET_CRYPTO_EcdsaPublicKey) * 8];
70 end = GNUNET_STRINGS_data_to_string ((const unsigned char *) z,
71 sizeof (struct GNUNET_CRYPTO_EcdsaPublicKey),
84 * Compares if two records are equal (ignoring flags such
85 * as authority, private and pending, but not relative vs.
86 * absolute expiration time).
90 * @return #GNUNET_YES if the records are equal or #GNUNET_NO if they are not
93 GNUNET_NAMESTORE_records_cmp (const struct GNUNET_NAMESTORE_RecordData *a,
94 const struct GNUNET_NAMESTORE_RecordData *b)
96 LOG (GNUNET_ERROR_TYPE_DEBUG,
97 "Comparing records\n");
98 if (a->record_type != b->record_type)
100 LOG (GNUNET_ERROR_TYPE_DEBUG,
101 "Record type %lu != %lu\n", a->record_type, b->record_type);
104 if ((a->expiration_time != b->expiration_time) &&
105 ((a->expiration_time != 0) && (b->expiration_time != 0)))
107 LOG (GNUNET_ERROR_TYPE_DEBUG,
108 "Expiration time %llu != %llu\n",
113 if ((a->flags & GNUNET_NAMESTORE_RF_RCMP_FLAGS)
114 != (b->flags & GNUNET_NAMESTORE_RF_RCMP_FLAGS))
116 LOG (GNUNET_ERROR_TYPE_DEBUG,
117 "Flags %lu (%lu) != %lu (%lu)\n", a->flags,
118 a->flags & GNUNET_NAMESTORE_RF_RCMP_FLAGS, b->flags,
119 b->flags & GNUNET_NAMESTORE_RF_RCMP_FLAGS);
122 if (a->data_size != b->data_size)
124 LOG (GNUNET_ERROR_TYPE_DEBUG,
125 "Data size %lu != %lu\n",
130 if (0 != memcmp (a->data, b->data, a->data_size))
132 LOG (GNUNET_ERROR_TYPE_DEBUG,
133 "Data contents do not match\n");
136 LOG (GNUNET_ERROR_TYPE_DEBUG,
137 "Records are equal\n");
143 * Returns the expiration time of the given block of records. The block
144 * expiration time is the expiration time of the record with smallest
147 * @param rd_count number of records given in @a rd
148 * @param rd array of records
149 * @return absolute expiration time
151 struct GNUNET_TIME_Absolute
152 GNUNET_NAMESTORE_record_get_expiration_time (unsigned int rd_count,
153 const struct GNUNET_NAMESTORE_RecordData *rd)
156 struct GNUNET_TIME_Absolute expire;
157 struct GNUNET_TIME_Absolute at;
158 struct GNUNET_TIME_Relative rt;
161 return GNUNET_TIME_UNIT_ZERO_ABS;
162 expire = GNUNET_TIME_UNIT_FOREVER_ABS;
163 for (c = 0; c < rd_count; c++)
165 if (0 != (rd[c].flags & GNUNET_NAMESTORE_RF_RELATIVE_EXPIRATION))
167 rt.rel_value_us = rd[c].expiration_time;
168 at = GNUNET_TIME_relative_to_absolute (rt);
172 at.abs_value_us = rd[c].expiration_time;
174 expire = GNUNET_TIME_absolute_min (at, expire);
176 LOG (GNUNET_ERROR_TYPE_DEBUG,
177 "Determined expiration time for block with %u records to be %s\n",
179 GNUNET_STRINGS_absolute_time_to_string (expire));
185 * Test if a given record is expired.
187 * @return #GNUNET_YES if the record is expired,
191 GNUNET_NAMESTORE_is_expired (const struct GNUNET_NAMESTORE_RecordData *rd)
193 struct GNUNET_TIME_Absolute at;
195 if (0 != (rd->flags & GNUNET_NAMESTORE_RF_RELATIVE_EXPIRATION))
197 at.abs_value_us = rd->expiration_time;
198 return (0 == GNUNET_TIME_absolute_get_remaining (at).rel_value_us) ? GNUNET_YES : GNUNET_NO;
203 * Convert public key to the respective absolute domain name in the
205 * This is one of the very few calls in the entire API that is
208 * @param pkey a public key with a point on the eliptic curve
209 * @return string "X.zkey" where X is the public
210 * key in an encoding suitable for DNS labels.
213 GNUNET_NAMESTORE_pkey_to_zkey (const struct GNUNET_CRYPTO_EcdsaPublicKey *pkey)
215 static char ret[128];
218 pkeys = GNUNET_CRYPTO_ecdsa_public_key_to_string (pkey);
219 GNUNET_snprintf (ret,
229 * Convert an absolute domain name in the ".zkey" pTLD to the
230 * respective public key.
232 * @param zkey string "X.zkey" where X is the coordinates of the public
233 * key in an encoding suitable for DNS labels.
234 * @param pkey set to a public key on the eliptic curve
235 * @return #GNUNET_SYSERR if @a zkey has the wrong syntax
238 GNUNET_NAMESTORE_zkey_to_pkey (const char *zkey,
239 struct GNUNET_CRYPTO_EcdsaPublicKey *pkey)
245 cpy = GNUNET_strdup (zkey);
247 if (NULL == (dot = strchr (x, (int) '.')))
250 if (0 != strcasecmp (dot + 1,
255 GNUNET_CRYPTO_ecdsa_public_key_from_string (x,
263 return GNUNET_SYSERR;
267 /* end of gnsrecord_misc.c */