forked from ElektraInitiative/libelektra
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathhash.c
123 lines (107 loc) · 3.83 KB
/
hash.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
/**
* @file
*
* @brief Provides functions to hash Elektra data structures.
*
* @copyright BSD License (see LICENSE.md or https://www.libelektra.org)
*/
#include <kdb.h>
#include <kdbease.h>
#include <kdberrors.h>
#include <kdbtypes.h>
#include <stdio.h>
#include <string.h>
#include "sha-256.h"
static void hash_to_string (char string[65], const uint8_t hash[32]);
/**
* Calculate a specification token for the KeySet of an application.
*
* The KeySet of an application is identified as all keys below the applications root key.
*
* @pre The parentKey must have the correct namespace. E.g. If only keys from the spec:/ should be considered for the token calculation,
* pass a key with KEY_NS_SPEC.
*
* @note Array parent key's (e.g., `/format/#`) are ignored for the token. See inline documentation below for rationale.
*
* @param hash_string A string. After successful execution this will contain the hash as hex-string.
* @param ks The KeySet for the application.
* @param parentKey The Key below which all the relevant keys are. Keys that are not below @p parentKey are ignored. The key's namespace is
* important (see preconditions)
* @retval false If an error occurred.
* @retval true If the computation was successful.
*/
kdb_boolean_t calculateSpecificationToken (char hash_string[65], KeySet * ks, Key * parentKey)
{
if (parentKey == NULL)
{
// Can't set error to parentKey when it is null.
return false;
}
if (hash_string == NULL)
{
ELEKTRA_SET_INTERNAL_ERROR (parentKey, "Param hash_string was NULL");
return false;
}
if (ks == NULL)
{
ELEKTRA_SET_INTERNAL_ERROR (parentKey, "Param ks was NULL");
return false;
}
// Initialize sha_256 for streaming
uint8_t hash[SIZE_OF_SHA_256_HASH];
struct Sha_256 sha_256;
sha_256_init (&sha_256, hash);
// Duplicate ks, then cut out parentKey and all keys below. These are the ones we take into account for token calculation.
KeySet * dupKs = ksDup (ks);
KeySet * cutKs = ksCut (dupKs, parentKey);
/**
* Loop through all keys relevant for token calculation.
*/
for (elektraCursor it = 0; it < ksGetSize (cutKs); ++it)
{
Key * currentKey = ksAtCursor (cutKs, it);
/**
* Ignore array parents for token calculation.
* Rationale: There is a bug in the spec plugin that is triggered on changing the size of an array.
* It leads to array parents vanishing from the spec namespace and thus a different token.
* See https://github.com/ElektraInitiative/libelektra/issues/4061
*/
if (strcmp (keyBaseName (currentKey), "#") == 0)
{
continue;
}
/**
* Include NULL teminator in this and all following key/value strings, to avoid the following bug:
* https://github.com/ElektraInitiative/libelektra/issues/4110
*/
sha_256_write (&sha_256, keyName (currentKey), keyGetNameSize (currentKey));
// Note: The value of the key itself is not relevant / part of specification. Only the key's name + its metadata!
KeySet * currentMetaKeys = keyMeta (currentKey);
// Feed name + values from meta keys into sha_256_write().
for (elektraCursor metaIt = 0; metaIt < ksGetSize (currentMetaKeys); metaIt++)
{
Key * currentMetaKey = ksAtCursor (currentMetaKeys, metaIt);
sha_256_write (&sha_256, keyName (currentMetaKey), keyGetNameSize (currentMetaKey));
sha_256_write (&sha_256, keyString (currentMetaKey), keyGetValueSize (currentMetaKey));
}
}
sha_256_close (&sha_256);
hash_to_string (hash_string, hash);
ksDel (dupKs);
ksDel (cutKs);
return true;
}
/**
* Convert hash array to hex string
* Copied from https://github.com/amosnier/sha-2/blob/f0d7baf076207b943649e68946049059f018c10b/test.c
* @param string A string array of length 65 (64 characters + \0)
* @param hash The input hash array
*/
static void hash_to_string (char string[65], const uint8_t hash[32])
{
size_t i;
for (i = 0; i < 32; i++)
{
string += sprintf (string, "%02x", hash[i]);
}
}