Generate a set of unique keys that can be validated without keeping a white-list












1














I need to generate sets of 10 bytes unique IDs. These sets can be quite big (i.e. 10000 values) and are checked by a limited-memory device for validity. So someone enters one of the IDs in the device, and the device should be able to discern whether the ID is genuine (generated by me).

The basic way would be to store in the device's memory the same set of IDs and check against the list, but I can't use all of that memory.

A second way I thought would be to use a CRC or a hash function: for instance all the IDs whose CRC is X are enabled. The problem here is that I should iterate through all the possible combination of IDs to find the ones giving the right CRC.

Ideally, I would like to find one/two functions that work like this:



 uint8_t * generate_ID(uint16_t index);
bool is_valid validate_key(uint8_t * ID);
//optional
uint16_t index find_index(uint8_t * ID);

//example
//generate id value from index 0
uint8_t ID[10] = generate_ID(0)
//id is now {0x0a, 0x0b, 0x0c, 0x0d, 0x0e, 0x0f, 0x0e, 0x0d, 0x0c, 0x0b}

bool is_valid = validate_key(ID);
//is_valid is True
uint16_t index = find_index(ID);
//index is now 0
ID[0] = 0xff; //change ID with random value
is_valid = validate_key(ID);
//is_valid is now False

//BONUS: use also a "seed" value, so that I can differentiate through sets of ids:
uint8_t * generate_ID(uint16_t index, uint16_t seed);
bool is_valid validate_key(uint8_t * ID, uint16_t seed);


find_index() is optional because once I know the key is valid I can simply iterate through all indexes to find the matching index.

Basically, the generate_ID() function should be complex enough so that it won't be easily guessable if not knowing a decent amount of ID, yet able to be computed on an embedded CPU with limited horsepower (Cortex M0)










share|improve this question






















  • Are the symbols numeric (10 symbols) or alpha-numeric (36 symbols if fixed case, 62 if mixed case, 64 if you allow two special characters plus fixed characters). Assuming 64 values per symbol, then 10 symbols = 2^60 possible values, and say you allow 2^14 = 16384 of them, then you could use a 46 bit CRC. It wouldn't be that secure if someone knew it was based solely on a CRC. Using some xor pattern on the data could help a bit.
    – rcgldr
    Nov 17 at 1:39
















1














I need to generate sets of 10 bytes unique IDs. These sets can be quite big (i.e. 10000 values) and are checked by a limited-memory device for validity. So someone enters one of the IDs in the device, and the device should be able to discern whether the ID is genuine (generated by me).

The basic way would be to store in the device's memory the same set of IDs and check against the list, but I can't use all of that memory.

A second way I thought would be to use a CRC or a hash function: for instance all the IDs whose CRC is X are enabled. The problem here is that I should iterate through all the possible combination of IDs to find the ones giving the right CRC.

Ideally, I would like to find one/two functions that work like this:



 uint8_t * generate_ID(uint16_t index);
bool is_valid validate_key(uint8_t * ID);
//optional
uint16_t index find_index(uint8_t * ID);

//example
//generate id value from index 0
uint8_t ID[10] = generate_ID(0)
//id is now {0x0a, 0x0b, 0x0c, 0x0d, 0x0e, 0x0f, 0x0e, 0x0d, 0x0c, 0x0b}

bool is_valid = validate_key(ID);
//is_valid is True
uint16_t index = find_index(ID);
//index is now 0
ID[0] = 0xff; //change ID with random value
is_valid = validate_key(ID);
//is_valid is now False

//BONUS: use also a "seed" value, so that I can differentiate through sets of ids:
uint8_t * generate_ID(uint16_t index, uint16_t seed);
bool is_valid validate_key(uint8_t * ID, uint16_t seed);


find_index() is optional because once I know the key is valid I can simply iterate through all indexes to find the matching index.

Basically, the generate_ID() function should be complex enough so that it won't be easily guessable if not knowing a decent amount of ID, yet able to be computed on an embedded CPU with limited horsepower (Cortex M0)










share|improve this question






















  • Are the symbols numeric (10 symbols) or alpha-numeric (36 symbols if fixed case, 62 if mixed case, 64 if you allow two special characters plus fixed characters). Assuming 64 values per symbol, then 10 symbols = 2^60 possible values, and say you allow 2^14 = 16384 of them, then you could use a 46 bit CRC. It wouldn't be that secure if someone knew it was based solely on a CRC. Using some xor pattern on the data could help a bit.
    – rcgldr
    Nov 17 at 1:39














1












1








1







I need to generate sets of 10 bytes unique IDs. These sets can be quite big (i.e. 10000 values) and are checked by a limited-memory device for validity. So someone enters one of the IDs in the device, and the device should be able to discern whether the ID is genuine (generated by me).

The basic way would be to store in the device's memory the same set of IDs and check against the list, but I can't use all of that memory.

A second way I thought would be to use a CRC or a hash function: for instance all the IDs whose CRC is X are enabled. The problem here is that I should iterate through all the possible combination of IDs to find the ones giving the right CRC.

Ideally, I would like to find one/two functions that work like this:



 uint8_t * generate_ID(uint16_t index);
bool is_valid validate_key(uint8_t * ID);
//optional
uint16_t index find_index(uint8_t * ID);

//example
//generate id value from index 0
uint8_t ID[10] = generate_ID(0)
//id is now {0x0a, 0x0b, 0x0c, 0x0d, 0x0e, 0x0f, 0x0e, 0x0d, 0x0c, 0x0b}

bool is_valid = validate_key(ID);
//is_valid is True
uint16_t index = find_index(ID);
//index is now 0
ID[0] = 0xff; //change ID with random value
is_valid = validate_key(ID);
//is_valid is now False

//BONUS: use also a "seed" value, so that I can differentiate through sets of ids:
uint8_t * generate_ID(uint16_t index, uint16_t seed);
bool is_valid validate_key(uint8_t * ID, uint16_t seed);


find_index() is optional because once I know the key is valid I can simply iterate through all indexes to find the matching index.

Basically, the generate_ID() function should be complex enough so that it won't be easily guessable if not knowing a decent amount of ID, yet able to be computed on an embedded CPU with limited horsepower (Cortex M0)










share|improve this question













I need to generate sets of 10 bytes unique IDs. These sets can be quite big (i.e. 10000 values) and are checked by a limited-memory device for validity. So someone enters one of the IDs in the device, and the device should be able to discern whether the ID is genuine (generated by me).

The basic way would be to store in the device's memory the same set of IDs and check against the list, but I can't use all of that memory.

A second way I thought would be to use a CRC or a hash function: for instance all the IDs whose CRC is X are enabled. The problem here is that I should iterate through all the possible combination of IDs to find the ones giving the right CRC.

Ideally, I would like to find one/two functions that work like this:



 uint8_t * generate_ID(uint16_t index);
bool is_valid validate_key(uint8_t * ID);
//optional
uint16_t index find_index(uint8_t * ID);

//example
//generate id value from index 0
uint8_t ID[10] = generate_ID(0)
//id is now {0x0a, 0x0b, 0x0c, 0x0d, 0x0e, 0x0f, 0x0e, 0x0d, 0x0c, 0x0b}

bool is_valid = validate_key(ID);
//is_valid is True
uint16_t index = find_index(ID);
//index is now 0
ID[0] = 0xff; //change ID with random value
is_valid = validate_key(ID);
//is_valid is now False

//BONUS: use also a "seed" value, so that I can differentiate through sets of ids:
uint8_t * generate_ID(uint16_t index, uint16_t seed);
bool is_valid validate_key(uint8_t * ID, uint16_t seed);


find_index() is optional because once I know the key is valid I can simply iterate through all indexes to find the matching index.

Basically, the generate_ID() function should be complex enough so that it won't be easily guessable if not knowing a decent amount of ID, yet able to be computed on an embedded CPU with limited horsepower (Cortex M0)







c hash key crc






share|improve this question













share|improve this question











share|improve this question




share|improve this question










asked Nov 16 at 13:29









Vitomakes

10810




10810












  • Are the symbols numeric (10 symbols) or alpha-numeric (36 symbols if fixed case, 62 if mixed case, 64 if you allow two special characters plus fixed characters). Assuming 64 values per symbol, then 10 symbols = 2^60 possible values, and say you allow 2^14 = 16384 of them, then you could use a 46 bit CRC. It wouldn't be that secure if someone knew it was based solely on a CRC. Using some xor pattern on the data could help a bit.
    – rcgldr
    Nov 17 at 1:39


















  • Are the symbols numeric (10 symbols) or alpha-numeric (36 symbols if fixed case, 62 if mixed case, 64 if you allow two special characters plus fixed characters). Assuming 64 values per symbol, then 10 symbols = 2^60 possible values, and say you allow 2^14 = 16384 of them, then you could use a 46 bit CRC. It wouldn't be that secure if someone knew it was based solely on a CRC. Using some xor pattern on the data could help a bit.
    – rcgldr
    Nov 17 at 1:39
















Are the symbols numeric (10 symbols) or alpha-numeric (36 symbols if fixed case, 62 if mixed case, 64 if you allow two special characters plus fixed characters). Assuming 64 values per symbol, then 10 symbols = 2^60 possible values, and say you allow 2^14 = 16384 of them, then you could use a 46 bit CRC. It wouldn't be that secure if someone knew it was based solely on a CRC. Using some xor pattern on the data could help a bit.
– rcgldr
Nov 17 at 1:39




Are the symbols numeric (10 symbols) or alpha-numeric (36 symbols if fixed case, 62 if mixed case, 64 if you allow two special characters plus fixed characters). Assuming 64 values per symbol, then 10 symbols = 2^60 possible values, and say you allow 2^14 = 16384 of them, then you could use a 46 bit CRC. It wouldn't be that secure if someone knew it was based solely on a CRC. Using some xor pattern on the data could help a bit.
– rcgldr
Nov 17 at 1:39












2 Answers
2






active

oldest

votes


















2














A 10 byte key is not enough to make anything secure.



You need a secure hash function such as SHA2-256 whose output is 32-byte in length. SHA2 can easily be implemented on most systems.



Your key needs two parts:



[text + hash]


The first part is like a "username" and the second part is like a "password"



You also need a "secret key". This key is an array of bytes which is stored in your software. You then add the "secret key" to your "username". Find the SHA2 hash for the resulting string. Now you have an output that is the length of original text + 32 bytes for the hash.



You can use this key as a unique verifiable ID.



To test they key's authenticity, take the "username" part and add your secret key. Take SHA2 of that string, the result should match "password"



If secrecy and uniqueness is not a big issue then you can use MD5 whose output is 16 bytes. Change plain text to binary so it can store more information in fewer bytes, and your final key will be only 20 bytes. You could cut that down a bit more but reducing to 10 bytes is not recommended.



Here is an example. I used SHA2 implementation from this link:
https://github.com/B-Con/crypto-algorithms (I am not sure if it works on a big-endian machine)



Any SHA2 implementation should work.



void sha2(BYTE* dst, const BYTE* src, int len)
{
SHA256_CTX ctx;
sha256_init(&ctx);
sha256_update(&ctx, (const BYTE*)src, len);
sha256_final(&ctx, (BYTE*)dst);
}

void create_verifiable_id(const BYTE* source, BYTE *uid)
{
BYTE hash[32];
sha2(hash, source, ID_SIZE);

//combine source + hash
memcpy(uid, source, ID_SIZE);
memcpy(uid + ID_SIZE, hash, 32);
}

int test_verfiable_id(const BYTE *uid)
{
BYTE hash[32];
sha2(hash, uid, ID_SIZE);

//hash should match the second part of uid
return memcmp(hash, uid + ID_SIZE, 32) == 0;
}

int main(void)
{
//use a number from 0 to 0xFFFFFFFF, store in buf (4 bytes)
//this is the "plain text" portion
int number = 0x12345678;
BYTE buf[ID_SIZE];
for(int i = 0; i < sizeof(buf); i++)
{
buf[i] = number & 0xFF;
number >>= 8;
}

//add sha2 to "plain text" to make verifiable id
BYTE verifiable_id[32 + ID_SIZE];
create_verifiable_id(buf, verifiable_id);

printf("UID as hex string:n");
for(int i = 0; i < 32 + ID_SIZE; i++)
printf("%02X", verifiable_id[i] & 0xFF);
printf("n");

printf("Test (should succeed): %dn", test_verfiable_id(verifiable_id));

//change verifiable_id and test it again
verifiable_id[0]++;
printf("Test (should fail): %dn", test_verfiable_id(verifiable_id));
return 0;
}





share|improve this answer























  • that's a good idea! Do you think I can use the same approach with AES128 which I have hardware functions for? Uniqueness is important, I don't want two IDs to collide, security is a bit less important.
    – Vitomakes
    Nov 18 at 7:52








  • 1




    I wouldn't know how to use AES in that way. The encryption blocks are unrelated so you have to use simple ECB mode which is weak. Also AES is a 2-way encryption, whereas hash functions are 1-way. MD5 is vulnerable to attack but it's better than AES in this case. SHA2 is better, it is used in password verification which is more applicable here.
    – Barmak Shemirani
    Nov 18 at 8:25










  • @Vitomakes, making a hash function from an encryption algorithm is not difficult, you have only to encrypt the source text into blocks of fixed size and then XOR all of them together (this is an example of just one way of doing) with a block size equal to the size of the hash you want. But what are you using that only supports AES, and no hashing at all? To avoid colliding you need large keys, you cannot avoid it completely, but on a probabilistic way (the space of source texts is always bigger than the space of hash keys)
    – Luis Colorado
    Nov 20 at 7:23










  • @LuisColorado AES is just "easier" because I have hardware peripherals for that on my CPU and I don't need to code anything at all. BTW the basic AES block is 16 bytes, so if I make the ID longer (exactly 16 bytes), I can just use 1 whole block with no risk of collision, right?
    – Vitomakes
    Nov 20 at 9:28










  • @Vitomakes, you can use e.g. a 32 bytes blocksize (two blocks of AES code, or even better, use a 30 byte block size and lose two bytes from each AES buffer, that will give less coherency between AES buffers) and get wider hash space, and you'll get less chance to collisions. Do you actually have AES by hardware but no hash function at all? strange.
    – Luis Colorado
    Nov 20 at 9:34



















0














One very simple way to do this is with a modular multiplicative inverse, as I describe in my blog post here: http://blog.mischel.com/2017/06/20/how-to-generate-random-looking-keys/.



The idea is that you map the numbers from 1 to some number x so that each number generates a unique value within that same range. So, for example, the mapping might be:



1 -> 9875
2 -> 362
3 -> 5247
...


This is a reversible calculation. So if f(1) => 9875, then g(9875) => 1.



In the code shown in the blog post, you can change the mapping by modifying the x and m values.



If you want the keys to be alphanumeric, then you need to encode them after generating the integer. And then you'll have to decode them back to an integer after the user enters it and before you try to validate.



So, for your validation, set m to a very large number. Set x appropriately, preferably a prime number larger than m. Generate the first 10,000 keys using those values. On the device that is supposed to validate these numbers, just supply the x and m values, and the maximum index (i.e. 10,000).



So a user enters the key they were given. You run the reverse of the key generation and get a number between 1 and 10,000. You know that the number is valid. If your reverse calculation returns a number that's less than 1 or greater than 10,000, then that key isn't valid.



You could extend this to multiple devices by just giving each device the start and end values that it considers valid. The reverse key calculation is the same, regardless.



This technique ensures uniqueness. Security is ... mostly through obscurity. If somebody knows the algorithm being used, including the x and m values, and knows the range of numbers that a device is set to accept, then he could generate keys to defeat the system. Whether that's a problem is something only you can answer. What is the risk of somebody trying to defeat your system, and what is the cost if they succeed?






share|improve this answer





















    Your Answer






    StackExchange.ifUsing("editor", function () {
    StackExchange.using("externalEditor", function () {
    StackExchange.using("snippets", function () {
    StackExchange.snippets.init();
    });
    });
    }, "code-snippets");

    StackExchange.ready(function() {
    var channelOptions = {
    tags: "".split(" "),
    id: "1"
    };
    initTagRenderer("".split(" "), "".split(" "), channelOptions);

    StackExchange.using("externalEditor", function() {
    // Have to fire editor after snippets, if snippets enabled
    if (StackExchange.settings.snippets.snippetsEnabled) {
    StackExchange.using("snippets", function() {
    createEditor();
    });
    }
    else {
    createEditor();
    }
    });

    function createEditor() {
    StackExchange.prepareEditor({
    heartbeatType: 'answer',
    autoActivateHeartbeat: false,
    convertImagesToLinks: true,
    noModals: true,
    showLowRepImageUploadWarning: true,
    reputationToPostImages: 10,
    bindNavPrevention: true,
    postfix: "",
    imageUploader: {
    brandingHtml: "Powered by u003ca class="icon-imgur-white" href="https://imgur.com/"u003eu003c/au003e",
    contentPolicyHtml: "User contributions licensed under u003ca href="https://creativecommons.org/licenses/by-sa/3.0/"u003ecc by-sa 3.0 with attribution requiredu003c/au003e u003ca href="https://stackoverflow.com/legal/content-policy"u003e(content policy)u003c/au003e",
    allowUrls: true
    },
    onDemand: true,
    discardSelector: ".discard-answer"
    ,immediatelyShowMarkdownHelp:true
    });


    }
    });














    draft saved

    draft discarded


















    StackExchange.ready(
    function () {
    StackExchange.openid.initPostLogin('.new-post-login', 'https%3a%2f%2fstackoverflow.com%2fquestions%2f53338861%2fgenerate-a-set-of-unique-keys-that-can-be-validated-without-keeping-a-white-list%23new-answer', 'question_page');
    }
    );

    Post as a guest















    Required, but never shown

























    2 Answers
    2






    active

    oldest

    votes








    2 Answers
    2






    active

    oldest

    votes









    active

    oldest

    votes






    active

    oldest

    votes









    2














    A 10 byte key is not enough to make anything secure.



    You need a secure hash function such as SHA2-256 whose output is 32-byte in length. SHA2 can easily be implemented on most systems.



    Your key needs two parts:



    [text + hash]


    The first part is like a "username" and the second part is like a "password"



    You also need a "secret key". This key is an array of bytes which is stored in your software. You then add the "secret key" to your "username". Find the SHA2 hash for the resulting string. Now you have an output that is the length of original text + 32 bytes for the hash.



    You can use this key as a unique verifiable ID.



    To test they key's authenticity, take the "username" part and add your secret key. Take SHA2 of that string, the result should match "password"



    If secrecy and uniqueness is not a big issue then you can use MD5 whose output is 16 bytes. Change plain text to binary so it can store more information in fewer bytes, and your final key will be only 20 bytes. You could cut that down a bit more but reducing to 10 bytes is not recommended.



    Here is an example. I used SHA2 implementation from this link:
    https://github.com/B-Con/crypto-algorithms (I am not sure if it works on a big-endian machine)



    Any SHA2 implementation should work.



    void sha2(BYTE* dst, const BYTE* src, int len)
    {
    SHA256_CTX ctx;
    sha256_init(&ctx);
    sha256_update(&ctx, (const BYTE*)src, len);
    sha256_final(&ctx, (BYTE*)dst);
    }

    void create_verifiable_id(const BYTE* source, BYTE *uid)
    {
    BYTE hash[32];
    sha2(hash, source, ID_SIZE);

    //combine source + hash
    memcpy(uid, source, ID_SIZE);
    memcpy(uid + ID_SIZE, hash, 32);
    }

    int test_verfiable_id(const BYTE *uid)
    {
    BYTE hash[32];
    sha2(hash, uid, ID_SIZE);

    //hash should match the second part of uid
    return memcmp(hash, uid + ID_SIZE, 32) == 0;
    }

    int main(void)
    {
    //use a number from 0 to 0xFFFFFFFF, store in buf (4 bytes)
    //this is the "plain text" portion
    int number = 0x12345678;
    BYTE buf[ID_SIZE];
    for(int i = 0; i < sizeof(buf); i++)
    {
    buf[i] = number & 0xFF;
    number >>= 8;
    }

    //add sha2 to "plain text" to make verifiable id
    BYTE verifiable_id[32 + ID_SIZE];
    create_verifiable_id(buf, verifiable_id);

    printf("UID as hex string:n");
    for(int i = 0; i < 32 + ID_SIZE; i++)
    printf("%02X", verifiable_id[i] & 0xFF);
    printf("n");

    printf("Test (should succeed): %dn", test_verfiable_id(verifiable_id));

    //change verifiable_id and test it again
    verifiable_id[0]++;
    printf("Test (should fail): %dn", test_verfiable_id(verifiable_id));
    return 0;
    }





    share|improve this answer























    • that's a good idea! Do you think I can use the same approach with AES128 which I have hardware functions for? Uniqueness is important, I don't want two IDs to collide, security is a bit less important.
      – Vitomakes
      Nov 18 at 7:52








    • 1




      I wouldn't know how to use AES in that way. The encryption blocks are unrelated so you have to use simple ECB mode which is weak. Also AES is a 2-way encryption, whereas hash functions are 1-way. MD5 is vulnerable to attack but it's better than AES in this case. SHA2 is better, it is used in password verification which is more applicable here.
      – Barmak Shemirani
      Nov 18 at 8:25










    • @Vitomakes, making a hash function from an encryption algorithm is not difficult, you have only to encrypt the source text into blocks of fixed size and then XOR all of them together (this is an example of just one way of doing) with a block size equal to the size of the hash you want. But what are you using that only supports AES, and no hashing at all? To avoid colliding you need large keys, you cannot avoid it completely, but on a probabilistic way (the space of source texts is always bigger than the space of hash keys)
      – Luis Colorado
      Nov 20 at 7:23










    • @LuisColorado AES is just "easier" because I have hardware peripherals for that on my CPU and I don't need to code anything at all. BTW the basic AES block is 16 bytes, so if I make the ID longer (exactly 16 bytes), I can just use 1 whole block with no risk of collision, right?
      – Vitomakes
      Nov 20 at 9:28










    • @Vitomakes, you can use e.g. a 32 bytes blocksize (two blocks of AES code, or even better, use a 30 byte block size and lose two bytes from each AES buffer, that will give less coherency between AES buffers) and get wider hash space, and you'll get less chance to collisions. Do you actually have AES by hardware but no hash function at all? strange.
      – Luis Colorado
      Nov 20 at 9:34
















    2














    A 10 byte key is not enough to make anything secure.



    You need a secure hash function such as SHA2-256 whose output is 32-byte in length. SHA2 can easily be implemented on most systems.



    Your key needs two parts:



    [text + hash]


    The first part is like a "username" and the second part is like a "password"



    You also need a "secret key". This key is an array of bytes which is stored in your software. You then add the "secret key" to your "username". Find the SHA2 hash for the resulting string. Now you have an output that is the length of original text + 32 bytes for the hash.



    You can use this key as a unique verifiable ID.



    To test they key's authenticity, take the "username" part and add your secret key. Take SHA2 of that string, the result should match "password"



    If secrecy and uniqueness is not a big issue then you can use MD5 whose output is 16 bytes. Change plain text to binary so it can store more information in fewer bytes, and your final key will be only 20 bytes. You could cut that down a bit more but reducing to 10 bytes is not recommended.



    Here is an example. I used SHA2 implementation from this link:
    https://github.com/B-Con/crypto-algorithms (I am not sure if it works on a big-endian machine)



    Any SHA2 implementation should work.



    void sha2(BYTE* dst, const BYTE* src, int len)
    {
    SHA256_CTX ctx;
    sha256_init(&ctx);
    sha256_update(&ctx, (const BYTE*)src, len);
    sha256_final(&ctx, (BYTE*)dst);
    }

    void create_verifiable_id(const BYTE* source, BYTE *uid)
    {
    BYTE hash[32];
    sha2(hash, source, ID_SIZE);

    //combine source + hash
    memcpy(uid, source, ID_SIZE);
    memcpy(uid + ID_SIZE, hash, 32);
    }

    int test_verfiable_id(const BYTE *uid)
    {
    BYTE hash[32];
    sha2(hash, uid, ID_SIZE);

    //hash should match the second part of uid
    return memcmp(hash, uid + ID_SIZE, 32) == 0;
    }

    int main(void)
    {
    //use a number from 0 to 0xFFFFFFFF, store in buf (4 bytes)
    //this is the "plain text" portion
    int number = 0x12345678;
    BYTE buf[ID_SIZE];
    for(int i = 0; i < sizeof(buf); i++)
    {
    buf[i] = number & 0xFF;
    number >>= 8;
    }

    //add sha2 to "plain text" to make verifiable id
    BYTE verifiable_id[32 + ID_SIZE];
    create_verifiable_id(buf, verifiable_id);

    printf("UID as hex string:n");
    for(int i = 0; i < 32 + ID_SIZE; i++)
    printf("%02X", verifiable_id[i] & 0xFF);
    printf("n");

    printf("Test (should succeed): %dn", test_verfiable_id(verifiable_id));

    //change verifiable_id and test it again
    verifiable_id[0]++;
    printf("Test (should fail): %dn", test_verfiable_id(verifiable_id));
    return 0;
    }





    share|improve this answer























    • that's a good idea! Do you think I can use the same approach with AES128 which I have hardware functions for? Uniqueness is important, I don't want two IDs to collide, security is a bit less important.
      – Vitomakes
      Nov 18 at 7:52








    • 1




      I wouldn't know how to use AES in that way. The encryption blocks are unrelated so you have to use simple ECB mode which is weak. Also AES is a 2-way encryption, whereas hash functions are 1-way. MD5 is vulnerable to attack but it's better than AES in this case. SHA2 is better, it is used in password verification which is more applicable here.
      – Barmak Shemirani
      Nov 18 at 8:25










    • @Vitomakes, making a hash function from an encryption algorithm is not difficult, you have only to encrypt the source text into blocks of fixed size and then XOR all of them together (this is an example of just one way of doing) with a block size equal to the size of the hash you want. But what are you using that only supports AES, and no hashing at all? To avoid colliding you need large keys, you cannot avoid it completely, but on a probabilistic way (the space of source texts is always bigger than the space of hash keys)
      – Luis Colorado
      Nov 20 at 7:23










    • @LuisColorado AES is just "easier" because I have hardware peripherals for that on my CPU and I don't need to code anything at all. BTW the basic AES block is 16 bytes, so if I make the ID longer (exactly 16 bytes), I can just use 1 whole block with no risk of collision, right?
      – Vitomakes
      Nov 20 at 9:28










    • @Vitomakes, you can use e.g. a 32 bytes blocksize (two blocks of AES code, or even better, use a 30 byte block size and lose two bytes from each AES buffer, that will give less coherency between AES buffers) and get wider hash space, and you'll get less chance to collisions. Do you actually have AES by hardware but no hash function at all? strange.
      – Luis Colorado
      Nov 20 at 9:34














    2












    2








    2






    A 10 byte key is not enough to make anything secure.



    You need a secure hash function such as SHA2-256 whose output is 32-byte in length. SHA2 can easily be implemented on most systems.



    Your key needs two parts:



    [text + hash]


    The first part is like a "username" and the second part is like a "password"



    You also need a "secret key". This key is an array of bytes which is stored in your software. You then add the "secret key" to your "username". Find the SHA2 hash for the resulting string. Now you have an output that is the length of original text + 32 bytes for the hash.



    You can use this key as a unique verifiable ID.



    To test they key's authenticity, take the "username" part and add your secret key. Take SHA2 of that string, the result should match "password"



    If secrecy and uniqueness is not a big issue then you can use MD5 whose output is 16 bytes. Change plain text to binary so it can store more information in fewer bytes, and your final key will be only 20 bytes. You could cut that down a bit more but reducing to 10 bytes is not recommended.



    Here is an example. I used SHA2 implementation from this link:
    https://github.com/B-Con/crypto-algorithms (I am not sure if it works on a big-endian machine)



    Any SHA2 implementation should work.



    void sha2(BYTE* dst, const BYTE* src, int len)
    {
    SHA256_CTX ctx;
    sha256_init(&ctx);
    sha256_update(&ctx, (const BYTE*)src, len);
    sha256_final(&ctx, (BYTE*)dst);
    }

    void create_verifiable_id(const BYTE* source, BYTE *uid)
    {
    BYTE hash[32];
    sha2(hash, source, ID_SIZE);

    //combine source + hash
    memcpy(uid, source, ID_SIZE);
    memcpy(uid + ID_SIZE, hash, 32);
    }

    int test_verfiable_id(const BYTE *uid)
    {
    BYTE hash[32];
    sha2(hash, uid, ID_SIZE);

    //hash should match the second part of uid
    return memcmp(hash, uid + ID_SIZE, 32) == 0;
    }

    int main(void)
    {
    //use a number from 0 to 0xFFFFFFFF, store in buf (4 bytes)
    //this is the "plain text" portion
    int number = 0x12345678;
    BYTE buf[ID_SIZE];
    for(int i = 0; i < sizeof(buf); i++)
    {
    buf[i] = number & 0xFF;
    number >>= 8;
    }

    //add sha2 to "plain text" to make verifiable id
    BYTE verifiable_id[32 + ID_SIZE];
    create_verifiable_id(buf, verifiable_id);

    printf("UID as hex string:n");
    for(int i = 0; i < 32 + ID_SIZE; i++)
    printf("%02X", verifiable_id[i] & 0xFF);
    printf("n");

    printf("Test (should succeed): %dn", test_verfiable_id(verifiable_id));

    //change verifiable_id and test it again
    verifiable_id[0]++;
    printf("Test (should fail): %dn", test_verfiable_id(verifiable_id));
    return 0;
    }





    share|improve this answer














    A 10 byte key is not enough to make anything secure.



    You need a secure hash function such as SHA2-256 whose output is 32-byte in length. SHA2 can easily be implemented on most systems.



    Your key needs two parts:



    [text + hash]


    The first part is like a "username" and the second part is like a "password"



    You also need a "secret key". This key is an array of bytes which is stored in your software. You then add the "secret key" to your "username". Find the SHA2 hash for the resulting string. Now you have an output that is the length of original text + 32 bytes for the hash.



    You can use this key as a unique verifiable ID.



    To test they key's authenticity, take the "username" part and add your secret key. Take SHA2 of that string, the result should match "password"



    If secrecy and uniqueness is not a big issue then you can use MD5 whose output is 16 bytes. Change plain text to binary so it can store more information in fewer bytes, and your final key will be only 20 bytes. You could cut that down a bit more but reducing to 10 bytes is not recommended.



    Here is an example. I used SHA2 implementation from this link:
    https://github.com/B-Con/crypto-algorithms (I am not sure if it works on a big-endian machine)



    Any SHA2 implementation should work.



    void sha2(BYTE* dst, const BYTE* src, int len)
    {
    SHA256_CTX ctx;
    sha256_init(&ctx);
    sha256_update(&ctx, (const BYTE*)src, len);
    sha256_final(&ctx, (BYTE*)dst);
    }

    void create_verifiable_id(const BYTE* source, BYTE *uid)
    {
    BYTE hash[32];
    sha2(hash, source, ID_SIZE);

    //combine source + hash
    memcpy(uid, source, ID_SIZE);
    memcpy(uid + ID_SIZE, hash, 32);
    }

    int test_verfiable_id(const BYTE *uid)
    {
    BYTE hash[32];
    sha2(hash, uid, ID_SIZE);

    //hash should match the second part of uid
    return memcmp(hash, uid + ID_SIZE, 32) == 0;
    }

    int main(void)
    {
    //use a number from 0 to 0xFFFFFFFF, store in buf (4 bytes)
    //this is the "plain text" portion
    int number = 0x12345678;
    BYTE buf[ID_SIZE];
    for(int i = 0; i < sizeof(buf); i++)
    {
    buf[i] = number & 0xFF;
    number >>= 8;
    }

    //add sha2 to "plain text" to make verifiable id
    BYTE verifiable_id[32 + ID_SIZE];
    create_verifiable_id(buf, verifiable_id);

    printf("UID as hex string:n");
    for(int i = 0; i < 32 + ID_SIZE; i++)
    printf("%02X", verifiable_id[i] & 0xFF);
    printf("n");

    printf("Test (should succeed): %dn", test_verfiable_id(verifiable_id));

    //change verifiable_id and test it again
    verifiable_id[0]++;
    printf("Test (should fail): %dn", test_verfiable_id(verifiable_id));
    return 0;
    }






    share|improve this answer














    share|improve this answer



    share|improve this answer








    edited Nov 21 at 6:53

























    answered Nov 16 at 23:54









    Barmak Shemirani

    20.8k42045




    20.8k42045












    • that's a good idea! Do you think I can use the same approach with AES128 which I have hardware functions for? Uniqueness is important, I don't want two IDs to collide, security is a bit less important.
      – Vitomakes
      Nov 18 at 7:52








    • 1




      I wouldn't know how to use AES in that way. The encryption blocks are unrelated so you have to use simple ECB mode which is weak. Also AES is a 2-way encryption, whereas hash functions are 1-way. MD5 is vulnerable to attack but it's better than AES in this case. SHA2 is better, it is used in password verification which is more applicable here.
      – Barmak Shemirani
      Nov 18 at 8:25










    • @Vitomakes, making a hash function from an encryption algorithm is not difficult, you have only to encrypt the source text into blocks of fixed size and then XOR all of them together (this is an example of just one way of doing) with a block size equal to the size of the hash you want. But what are you using that only supports AES, and no hashing at all? To avoid colliding you need large keys, you cannot avoid it completely, but on a probabilistic way (the space of source texts is always bigger than the space of hash keys)
      – Luis Colorado
      Nov 20 at 7:23










    • @LuisColorado AES is just "easier" because I have hardware peripherals for that on my CPU and I don't need to code anything at all. BTW the basic AES block is 16 bytes, so if I make the ID longer (exactly 16 bytes), I can just use 1 whole block with no risk of collision, right?
      – Vitomakes
      Nov 20 at 9:28










    • @Vitomakes, you can use e.g. a 32 bytes blocksize (two blocks of AES code, or even better, use a 30 byte block size and lose two bytes from each AES buffer, that will give less coherency between AES buffers) and get wider hash space, and you'll get less chance to collisions. Do you actually have AES by hardware but no hash function at all? strange.
      – Luis Colorado
      Nov 20 at 9:34


















    • that's a good idea! Do you think I can use the same approach with AES128 which I have hardware functions for? Uniqueness is important, I don't want two IDs to collide, security is a bit less important.
      – Vitomakes
      Nov 18 at 7:52








    • 1




      I wouldn't know how to use AES in that way. The encryption blocks are unrelated so you have to use simple ECB mode which is weak. Also AES is a 2-way encryption, whereas hash functions are 1-way. MD5 is vulnerable to attack but it's better than AES in this case. SHA2 is better, it is used in password verification which is more applicable here.
      – Barmak Shemirani
      Nov 18 at 8:25










    • @Vitomakes, making a hash function from an encryption algorithm is not difficult, you have only to encrypt the source text into blocks of fixed size and then XOR all of them together (this is an example of just one way of doing) with a block size equal to the size of the hash you want. But what are you using that only supports AES, and no hashing at all? To avoid colliding you need large keys, you cannot avoid it completely, but on a probabilistic way (the space of source texts is always bigger than the space of hash keys)
      – Luis Colorado
      Nov 20 at 7:23










    • @LuisColorado AES is just "easier" because I have hardware peripherals for that on my CPU and I don't need to code anything at all. BTW the basic AES block is 16 bytes, so if I make the ID longer (exactly 16 bytes), I can just use 1 whole block with no risk of collision, right?
      – Vitomakes
      Nov 20 at 9:28










    • @Vitomakes, you can use e.g. a 32 bytes blocksize (two blocks of AES code, or even better, use a 30 byte block size and lose two bytes from each AES buffer, that will give less coherency between AES buffers) and get wider hash space, and you'll get less chance to collisions. Do you actually have AES by hardware but no hash function at all? strange.
      – Luis Colorado
      Nov 20 at 9:34
















    that's a good idea! Do you think I can use the same approach with AES128 which I have hardware functions for? Uniqueness is important, I don't want two IDs to collide, security is a bit less important.
    – Vitomakes
    Nov 18 at 7:52






    that's a good idea! Do you think I can use the same approach with AES128 which I have hardware functions for? Uniqueness is important, I don't want two IDs to collide, security is a bit less important.
    – Vitomakes
    Nov 18 at 7:52






    1




    1




    I wouldn't know how to use AES in that way. The encryption blocks are unrelated so you have to use simple ECB mode which is weak. Also AES is a 2-way encryption, whereas hash functions are 1-way. MD5 is vulnerable to attack but it's better than AES in this case. SHA2 is better, it is used in password verification which is more applicable here.
    – Barmak Shemirani
    Nov 18 at 8:25




    I wouldn't know how to use AES in that way. The encryption blocks are unrelated so you have to use simple ECB mode which is weak. Also AES is a 2-way encryption, whereas hash functions are 1-way. MD5 is vulnerable to attack but it's better than AES in this case. SHA2 is better, it is used in password verification which is more applicable here.
    – Barmak Shemirani
    Nov 18 at 8:25












    @Vitomakes, making a hash function from an encryption algorithm is not difficult, you have only to encrypt the source text into blocks of fixed size and then XOR all of them together (this is an example of just one way of doing) with a block size equal to the size of the hash you want. But what are you using that only supports AES, and no hashing at all? To avoid colliding you need large keys, you cannot avoid it completely, but on a probabilistic way (the space of source texts is always bigger than the space of hash keys)
    – Luis Colorado
    Nov 20 at 7:23




    @Vitomakes, making a hash function from an encryption algorithm is not difficult, you have only to encrypt the source text into blocks of fixed size and then XOR all of them together (this is an example of just one way of doing) with a block size equal to the size of the hash you want. But what are you using that only supports AES, and no hashing at all? To avoid colliding you need large keys, you cannot avoid it completely, but on a probabilistic way (the space of source texts is always bigger than the space of hash keys)
    – Luis Colorado
    Nov 20 at 7:23












    @LuisColorado AES is just "easier" because I have hardware peripherals for that on my CPU and I don't need to code anything at all. BTW the basic AES block is 16 bytes, so if I make the ID longer (exactly 16 bytes), I can just use 1 whole block with no risk of collision, right?
    – Vitomakes
    Nov 20 at 9:28




    @LuisColorado AES is just "easier" because I have hardware peripherals for that on my CPU and I don't need to code anything at all. BTW the basic AES block is 16 bytes, so if I make the ID longer (exactly 16 bytes), I can just use 1 whole block with no risk of collision, right?
    – Vitomakes
    Nov 20 at 9:28












    @Vitomakes, you can use e.g. a 32 bytes blocksize (two blocks of AES code, or even better, use a 30 byte block size and lose two bytes from each AES buffer, that will give less coherency between AES buffers) and get wider hash space, and you'll get less chance to collisions. Do you actually have AES by hardware but no hash function at all? strange.
    – Luis Colorado
    Nov 20 at 9:34




    @Vitomakes, you can use e.g. a 32 bytes blocksize (two blocks of AES code, or even better, use a 30 byte block size and lose two bytes from each AES buffer, that will give less coherency between AES buffers) and get wider hash space, and you'll get less chance to collisions. Do you actually have AES by hardware but no hash function at all? strange.
    – Luis Colorado
    Nov 20 at 9:34













    0














    One very simple way to do this is with a modular multiplicative inverse, as I describe in my blog post here: http://blog.mischel.com/2017/06/20/how-to-generate-random-looking-keys/.



    The idea is that you map the numbers from 1 to some number x so that each number generates a unique value within that same range. So, for example, the mapping might be:



    1 -> 9875
    2 -> 362
    3 -> 5247
    ...


    This is a reversible calculation. So if f(1) => 9875, then g(9875) => 1.



    In the code shown in the blog post, you can change the mapping by modifying the x and m values.



    If you want the keys to be alphanumeric, then you need to encode them after generating the integer. And then you'll have to decode them back to an integer after the user enters it and before you try to validate.



    So, for your validation, set m to a very large number. Set x appropriately, preferably a prime number larger than m. Generate the first 10,000 keys using those values. On the device that is supposed to validate these numbers, just supply the x and m values, and the maximum index (i.e. 10,000).



    So a user enters the key they were given. You run the reverse of the key generation and get a number between 1 and 10,000. You know that the number is valid. If your reverse calculation returns a number that's less than 1 or greater than 10,000, then that key isn't valid.



    You could extend this to multiple devices by just giving each device the start and end values that it considers valid. The reverse key calculation is the same, regardless.



    This technique ensures uniqueness. Security is ... mostly through obscurity. If somebody knows the algorithm being used, including the x and m values, and knows the range of numbers that a device is set to accept, then he could generate keys to defeat the system. Whether that's a problem is something only you can answer. What is the risk of somebody trying to defeat your system, and what is the cost if they succeed?






    share|improve this answer


























      0














      One very simple way to do this is with a modular multiplicative inverse, as I describe in my blog post here: http://blog.mischel.com/2017/06/20/how-to-generate-random-looking-keys/.



      The idea is that you map the numbers from 1 to some number x so that each number generates a unique value within that same range. So, for example, the mapping might be:



      1 -> 9875
      2 -> 362
      3 -> 5247
      ...


      This is a reversible calculation. So if f(1) => 9875, then g(9875) => 1.



      In the code shown in the blog post, you can change the mapping by modifying the x and m values.



      If you want the keys to be alphanumeric, then you need to encode them after generating the integer. And then you'll have to decode them back to an integer after the user enters it and before you try to validate.



      So, for your validation, set m to a very large number. Set x appropriately, preferably a prime number larger than m. Generate the first 10,000 keys using those values. On the device that is supposed to validate these numbers, just supply the x and m values, and the maximum index (i.e. 10,000).



      So a user enters the key they were given. You run the reverse of the key generation and get a number between 1 and 10,000. You know that the number is valid. If your reverse calculation returns a number that's less than 1 or greater than 10,000, then that key isn't valid.



      You could extend this to multiple devices by just giving each device the start and end values that it considers valid. The reverse key calculation is the same, regardless.



      This technique ensures uniqueness. Security is ... mostly through obscurity. If somebody knows the algorithm being used, including the x and m values, and knows the range of numbers that a device is set to accept, then he could generate keys to defeat the system. Whether that's a problem is something only you can answer. What is the risk of somebody trying to defeat your system, and what is the cost if they succeed?






      share|improve this answer
























        0












        0








        0






        One very simple way to do this is with a modular multiplicative inverse, as I describe in my blog post here: http://blog.mischel.com/2017/06/20/how-to-generate-random-looking-keys/.



        The idea is that you map the numbers from 1 to some number x so that each number generates a unique value within that same range. So, for example, the mapping might be:



        1 -> 9875
        2 -> 362
        3 -> 5247
        ...


        This is a reversible calculation. So if f(1) => 9875, then g(9875) => 1.



        In the code shown in the blog post, you can change the mapping by modifying the x and m values.



        If you want the keys to be alphanumeric, then you need to encode them after generating the integer. And then you'll have to decode them back to an integer after the user enters it and before you try to validate.



        So, for your validation, set m to a very large number. Set x appropriately, preferably a prime number larger than m. Generate the first 10,000 keys using those values. On the device that is supposed to validate these numbers, just supply the x and m values, and the maximum index (i.e. 10,000).



        So a user enters the key they were given. You run the reverse of the key generation and get a number between 1 and 10,000. You know that the number is valid. If your reverse calculation returns a number that's less than 1 or greater than 10,000, then that key isn't valid.



        You could extend this to multiple devices by just giving each device the start and end values that it considers valid. The reverse key calculation is the same, regardless.



        This technique ensures uniqueness. Security is ... mostly through obscurity. If somebody knows the algorithm being used, including the x and m values, and knows the range of numbers that a device is set to accept, then he could generate keys to defeat the system. Whether that's a problem is something only you can answer. What is the risk of somebody trying to defeat your system, and what is the cost if they succeed?






        share|improve this answer












        One very simple way to do this is with a modular multiplicative inverse, as I describe in my blog post here: http://blog.mischel.com/2017/06/20/how-to-generate-random-looking-keys/.



        The idea is that you map the numbers from 1 to some number x so that each number generates a unique value within that same range. So, for example, the mapping might be:



        1 -> 9875
        2 -> 362
        3 -> 5247
        ...


        This is a reversible calculation. So if f(1) => 9875, then g(9875) => 1.



        In the code shown in the blog post, you can change the mapping by modifying the x and m values.



        If you want the keys to be alphanumeric, then you need to encode them after generating the integer. And then you'll have to decode them back to an integer after the user enters it and before you try to validate.



        So, for your validation, set m to a very large number. Set x appropriately, preferably a prime number larger than m. Generate the first 10,000 keys using those values. On the device that is supposed to validate these numbers, just supply the x and m values, and the maximum index (i.e. 10,000).



        So a user enters the key they were given. You run the reverse of the key generation and get a number between 1 and 10,000. You know that the number is valid. If your reverse calculation returns a number that's less than 1 or greater than 10,000, then that key isn't valid.



        You could extend this to multiple devices by just giving each device the start and end values that it considers valid. The reverse key calculation is the same, regardless.



        This technique ensures uniqueness. Security is ... mostly through obscurity. If somebody knows the algorithm being used, including the x and m values, and knows the range of numbers that a device is set to accept, then he could generate keys to defeat the system. Whether that's a problem is something only you can answer. What is the risk of somebody trying to defeat your system, and what is the cost if they succeed?







        share|improve this answer












        share|improve this answer



        share|improve this answer










        answered Nov 21 at 19:36









        Jim Mischel

        106k12128247




        106k12128247






























            draft saved

            draft discarded




















































            Thanks for contributing an answer to Stack Overflow!


            • Please be sure to answer the question. Provide details and share your research!

            But avoid



            • Asking for help, clarification, or responding to other answers.

            • Making statements based on opinion; back them up with references or personal experience.


            To learn more, see our tips on writing great answers.





            Some of your past answers have not been well-received, and you're in danger of being blocked from answering.


            Please pay close attention to the following guidance:


            • Please be sure to answer the question. Provide details and share your research!

            But avoid



            • Asking for help, clarification, or responding to other answers.

            • Making statements based on opinion; back them up with references or personal experience.


            To learn more, see our tips on writing great answers.




            draft saved


            draft discarded














            StackExchange.ready(
            function () {
            StackExchange.openid.initPostLogin('.new-post-login', 'https%3a%2f%2fstackoverflow.com%2fquestions%2f53338861%2fgenerate-a-set-of-unique-keys-that-can-be-validated-without-keeping-a-white-list%23new-answer', 'question_page');
            }
            );

            Post as a guest















            Required, but never shown





















































            Required, but never shown














            Required, but never shown












            Required, but never shown







            Required, but never shown

































            Required, but never shown














            Required, but never shown












            Required, but never shown







            Required, but never shown







            Popular posts from this blog

            404 Error Contact Form 7 ajax form submitting

            How to know if a Active Directory user can login interactively

            Refactoring coordinates for Minecraft Pi buildings written in Python