Returns a random v4 UUID of the form 00000000-0000-4000-8000-000000000000.
-
Star
(1,094)
You must be signed in to star a gist -
Fork
(106)
You must be signed in to fork a gist
-
-
Save jed/982883 to your computer and use it in GitHub Desktop.
| function() { | |
| return( // return | |
| "" + // a concatentated string: | |
| 1e7 + // 10000000 + | |
| -1e3 + // -1000 + | |
| -4e3 + // -4000 + | |
| -8e3 + // -80000000 + | |
| -1e11 // -100000000000, | |
| ).replace( // replacing | |
| /1|0/g, // ones and zeroes with a | |
| function() { | |
| return( | |
| 0 | // floored | |
| Math.random() * 16 // random integer from 0 to 15 | |
| ).toString(16) // as a hex digit. | |
| } | |
| ) | |
| } |
| function(){return(""+1e7+-1e3+-4e3+-8e3+-1e11).replace(/1|0/g,function(){return(0|Math.random()*16).toString(16)})} |
| DO WHAT THE FUCK YOU WANT TO PUBLIC LICENSE | |
| Version 2, December 2004 | |
| Copyright (C) 2011 Jed Schmidt <http://jed.is> | |
| Everyone is permitted to copy and distribute verbatim or modified | |
| copies of this license document, and changing it is allowed as long | |
| as the name is changed. | |
| DO WHAT THE FUCK YOU WANT TO PUBLIC LICENSE | |
| TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION | |
| 0. You just DO WHAT THE FUCK YOU WANT TO. |
| { | |
| "name": "UUID", | |
| "description": "Generates random UUIDs", | |
| "keywords": [ | |
| "UUID", | |
| "ID", | |
| "random" | |
| ] | |
| } |
Who's reading this at 2018-11-02T09:07:56Z ?
[2019 CReW RePoRTiNG iN] ~~ like if you're still watching this epic gist in 2019 ~~ [2019 CReW RePoRTiNG iN]
Also, any resolution to the latest collision issues? Are people using this in production?
Beautiful snippet, btw.
Using this in production, several projects, no collisions yet. I've considered changing that Math.random to something Date.now() based if it does happen, but if it ain't broke...
This gist is a good example of the "Look at me, I'm a genius programmer who is so smart I can write unmaintainable code" disease.
@AndrWeisR code golfing is harmless fun and a useful exercise that teaches you the details of the language. Would I put this in a project when I'm trying to collaborate with others? of course not, but golfing isn't just "genius programmers showing off", it's also just programmers challenging each other to practice, improve and have 𝒇𝒖𝒏™. 🙃
A UUID generator is frequently treated as a black box by those who use one anyway. For such people, they'd rather have a black box with less code rather than a black box with more code, even if the code was the product of a nerdy online challenge. 🤷♂️
This gist is a good example of the "Look at me, I'm a genius programmer who is so smart I can write unmaintainable code" disease.
Look at me "I'm jealous" disease. Take it as a challenge. It's just a gist, chill.
Thank you for this. This is what I had to do for: TypeScript + ESLint
const getUuid = (a: string = ''): string => (
a
/* eslint-disable no-bitwise */
? ((Number(a) ^ Math.random() * 16) >> Number(a) / 4).toString(16)
: (`${1e7}-${1e3}-${4e3}-${8e3}-${1e11}`).replace(/[018]/g, getUuid)
);
getUuid();finally I found a place to learn some code 👍
Here's UUID v1. Doesn't fit in tweet 😒
// http://www.rfcreader.com/#rfc4122_line385 allows random instead of MAC address
// https://www.famkruithof.net/uuid/uuidgen
// https://realityripple.com/Tools/UnUUID/
// TODO: Refactor into generator function, so random clock ID stays uniform for the lifetime of application instance?
function uuid(c = 9999) {
const t = ((Date.now() + 12219292800000) * 1E4).toString(16)
const n = crypto.getRandomValues(new Uint8Array(6)).reduce((sum, x, i) => {
return sum + ((i === 0) ? x|1 : x).toString(16).padStart(2, '0')
}, '')
return `${t.slice(-8)}-${t.slice(-12, -8)}-1${t.slice(0, 3)}-${c}-${n}`
}
// Give me some examples
for (let i = 100; i--; console.log(uuid())) {}Here's an ES-updated take on @mindplay-dk's non-golf version:
const uuidv4 = () => {
const hex = [...Array(256).keys()]
.map(index => (index).toString(16).padStart(2, '0'));
const r = crypto.getRandomValues(new Uint8Array(16));
r[6] = (r[6] & 0x0f) | 0x40;
r[8] = (r[8] & 0x3f) | 0x80;
return [...r.entries()]
.map(([index, int]) => [4, 6, 8, 10].includes(index) ? `-${hex[int]}` : hex[int])
.join('');
};Has anyone written a version which doesn't use bitwise operators?
@jsejcksn you probably should hoist your hex table to the parent scope, to avoid recalculating every time. Also note (though you might not care, depending on your use-case) that I was deliberately avoiding loops in my version for performance reasons - using map makes for fewer lines, but much slower execution. 🙂
@mindplay-dk Thanks, it’s primarily for readability.
For max optimization, one of the other golfed versions could be used, or you could compile your code using an optimizing transpiler.
We have been using this in production on the web, and are seeing a lot of collisions with the versions using Math.random in certain browsers. If you are using any of these snippets on the web, definitely use one that uses crypto.getRandomValues! Even if you tested it on your system and did not see any collisions, some other system/browser combination may implement Math.random in a way that causes easy collisions.
It has been a huge treat to read this thread, so much positivity, creativity, knowledge and collaboration!
Hi, what is the latest version of this function?
The latest version I've got is this:
var uuid = () => ([1e7]+-1e3+-4e3+-8e3+-1e11).replace(/[018]/g,c =>(c^(window.crypto||window.msCrypto).getRandomValues(new Uint8Array(1))[0]&15>>c/4).toString(16));Is the OP not updating the code anymore? @phyreman how does your function differ?
@RobertoMachorro Just ensuring the random numbers come from a crypto source instead of Math.random(). It's definitely slower than molasses though, lol. I think the OP stopped once it hit the point of basically not being able to golf it much more and a spec-compliant version was posted.
Ok I don't even know how, but I typed in 'var konami' with github copilot and it gave me:
var konami = { // https://gist.github.com/jed/982883
"up": false,
"down": false,
"left": false,
"right": false,
"b": false,
"a": false,
"start": false,
}how
Can anyone explain the reasoning behind what seems rather arbitrary, in the replacing only 0s, 1s, and 8s with random hex digits, and not, for example, replacing all numbers with such digits? I'm sure I missed some detail that explains it.
IIRC the spec requires those digits to be certain values.
@jed thanks for the link. I browsed through that spec to see if I could find something that jumps out. I see lots of discussion of bits in fields and such, but nothing that obviously connects digits 0, 1, and 8 to needing to be replaced. In fact, the part you linked to specifically says:
Set all the other bits to randomly (or pseudo-randomly) chosen values.
In any case, appreciate having further info. My understanding is still incomplete, but it's less incomplete now. ;-)
Oh, wait... I think I maybe get more of it now... the code has stuff like 1e3 and 8e3, which are like 1000 and 8000... but it also has 4e3... so basically it's targeting all the 1s, 0s, and 8s for replacement, but leaving the 4 in the middle alone.
That's because it's not part of the spec, it's part of the genius of @subzey. Look at the string that's being replaced
0and1(1is used as the head of a segment to prevent truncation),8, which is special cased with the^operation to clamp it to the proper values, but not4, which cannot be replaced because it specifies the version in the spec.
right, you got it. the 1 could be any other digit, really, it's just a placeholder, but the 8 was selected for its bitwise operation properties.
(there's a walkthrough here, fwiw)
@jed thanks! :)
I was perusing the minified source of a website to see how they did something, came across this interesting way to create a GUID, and traced it back to here. I thought it was interestingly minified in the source I was looking at (with an insanely clever minifier!). Made me laugh when I saw that the original was golfed. 😆 A+ on the golfing, BTW! Clever minifier indeed--a clever human minifier. This was really fun to mentally chew on.
For funsies, here's a rough draft on a UUIDv7 generator:
[...Date.now().toString(16).padStart(12,0)].toSpliced(8,0,'-').join``+([-7e3]+-8e3+-1e11).replace(/[018]/g,b=>(b^crypto.rng(1)[0]%16>>b/4).toString(16))
@AdamRGrey of course 😊