Skip to content

Instantly share code, notes, and snippets.

@postspectacular
Last active June 2, 2022 17:22
Show Gist options
  • Star 0 You must be signed in to star a gist
  • Fork 0 You must be signed in to fork a gist
  • Save postspectacular/f85d1d21597f9c8824e97f0c6b97d34e to your computer and use it in GitHub Desktop.
Save postspectacular/f85d1d21597f9c8824e97f0c6b97d34e to your computer and use it in GitHub Desktop.
Lottery script for picking a number of weighted random addresses of FXHash token holders, import from a number of CSV files. These CSV files are per-project and each have the format: `address, numTokens`. Furthermore, the weight of the projects can be adjusted via the `PROJECT_WEIGHTS` object...
import { float, parseCSVSimpleFromString } from "@thi.ng/csv";
import { readText } from "@thi.ng/file-io";
import { weightedRandomKey } from "@thi.ng/random";
import {
assocObj,
keys,
map,
normFrequenciesAuto,
repeatedly,
transduce,
} from "@thi.ng/transducers";
/**
* Number of lottery winners
*/
const NUM_WINNERS = 150;
/**
* Project weights adjusted according to Twitter poll:
* https://twitter.com/toxi/status/1532095956514131969
*/
const PROJECT_WEIGHTS = {
"defrag-v1": 1,
"defrag-v2": 0.189,
"defrag-v3": 0.089,
quasiflock: 0.721,
};
/**
* Parse CSV into object of weights (addresses as keys, number of tokens held as values)
*
* CSV obtained from:
* https://fxcollectors.stroep.nl/?project=981#page=owners
*/
const csvWeightMap = (path) =>
assocObj(
parseCSVSimpleFromString(
{ header: false, cols: [true, float()] },
readText(path)
)
);
/**
* Load all project CSV files and construct nested object of addresses per
* project, i.e.
*
* ```
* {
* "defrag-v1": { "tz1A...": 20, "tz1B...": 5 },
* "defrag-v2": { "tz1C...": 3, "tz1D...": 11 },
* ...
* }
* ```
*/
const addresses = transduce(
map((id) => [id, csvWeightMap(id + ".csv")]),
assocObj(),
keys(PROJECT_WEIGHTS)
);
/**
* Create a function, if when called returns a weighted random project ID.
*/
const pickProject = weightedRandomKey(PROJECT_WEIGHTS);
/**
* Similar to above, but for picking actual addresses (grouped by project),
* weighted by number of tokens per address.
*/
const projectAddressPickers = transduce(
map((id) => [id, weightedRandomKey(addresses[id])]),
assocObj(),
keys(PROJECT_WEIGHTS)
);
/**
* Draw a random list of project IDs (weighted)
*/
const winnerProjects = [...repeatedly(pickProject, NUM_WINNERS)];
/**
* For each picked project ID, pick a weighted winner address
*/
const winnerAddresses = winnerProjects.map((id) => projectAddressPickers[id]());
/**
* Print (sorted) winners
*/
console.log(winnerAddresses.sort().join("\n"));
/**
* Print out histogram of project & address distributions
*/
console.log(normFrequenciesAuto(winnerProjects));
console.log(normFrequenciesAuto(winnerAddresses));
@postspectacular
Copy link
Author

Alphabetically sorted list of addresses and their allocated slots (for the C-SCAPE fxhash drop), randomly chosen via the above script (only reformatted as CSV):

address,slots
tz1aB4WhspBPfLA6X4qSQPZCDKQkmmGobrCk,1
tz1abTpHKkdo5YTM1DosZZVx9p8cjv4hMMTB,1
tz1aCn4HgS1s1MkdBpN644RLQbZdSqSbNu3P,2
tz1amcnQuDXY2pzBy23e4sV8qaxpRbQAY7W6,1
tz1aV3qdpig8XeHHa1zXRUZ5PJfed2e7NnXF,1
tz1b8gWg7Xa28LEuFzxcXXU9Se5K4smaVpyX,1
tz1c3DCCLEWn8G323g49Umvyg8aEVK6ZpkiF,1
tz1cbJ2fHK4Tv7yES7Tq9dc9k8gXArnk7DyE,1
tz1cdowmr8kiWZpN5cESMw19bx3Gamupvkji,2
tz1cNDNPnuBzKPcbSyTbASZCqt8uaUMjXPF3,1
tz1cWXP9EE31jYuoHCMJzPFSX4ui27yNFehn,1
tz1cxYoXMbdu443mMtKqMutsZESq7H78eMvA,1
tz1d7CeaBZbETihDsyniHgEZDnEBajyNQBkX,1
tz1d82CMQZ58iK3LdME46a9H8tC87347Y6R1,1
tz1d8nbh2g2XfcCLzChJEybesNWfSmPFfUTR,1
tz1d8uBvibwk2Ct12pcSMg4D9n8PHez5pfPo,1
tz1dDSuGcw1YeARWpKaXt3Vp4htD8szbPCdY,1
tz1eHtSYDmFePHvZYHQZvQLk1kpvfcMFdLeJ,2
tz1f4R7mbvLRELULmDQxc52bra4uaEwkfRjc,1
tz1fgQgnq7itzJXL3Nvx6GLsyNfrv2ofDt9i,1
tz1fhd4k1jSJRRcNVbkMwVzkKEhX3kzApsfk,2
tz1fTC3BXuCjQx1AJp6JNSL5GTuQ9yK9Q2df,1
tz1g5XT1mitUqswejurB1WX4TnYGSAnMqVY7,1
tz1gBXG9fg8RMDH69KfKqwoTH5sFDmzt5yzm,2
tz1gigsw3HPNSjpRdaZ6QzTUckg48fopSAGj,1
tz1gMfctX4hBNpkUoE7RcYPBhNc1hpHddqh4,1
tz1gQyvkV8ofzcRjUhancM47wK8MuHz6Thzf,1
tz1h2uK9nCke3AW6AXrcGyK78C2nsC4XTHui,1
tz1hhG3vN2NCoPZepvEbKHBFQ5denwNJwHXP,1
tz1hm31gL4DXUeoaKsHCeWW91JFPrdBYgnnD,1
tz1huw6ywHzBv5Y1s6X1SW4E3JGjqJjcWW79,1
tz1i8QBJUiy3StRLzQ5ukt2J46j2bBt3rsUB,1
tz1iaVFySeteNqer5NcmNaxUdBW4bbrtNakb,2
tz1iB8Ayco6yPe2nBUJ8kbcF7CitAW8Aid74,1
tz1iCi9BdMEE36L8DjaD9dQjwqLnXFALHXeZ,2
tz1icoVXYJwDEmqXbQ17gzH6JPXrSpYb5qhd,2
tz1iEN3rnLvJN8Ze8Gq21CSwVbuDKn6UQWne,1
tz1ikMAhZYa5yon9CTwPQQrcPCPfES2d8u7N,2
tz1iocTPmx2ZzPiqLBfz8hsu9fTGdLz4NfXA,2
tz1ixR1g7FQrftBLK3aLk1kPcmvA9Sh9XsbF,3
tz1iZ9yHJzgvC4oBvHsrNmtL8GHPqtwqshAi,1
tz1KeV4HFbVPyNBeZZFUG9sGjaKm12rawhuM,1
tz1KjNhhxK96mUMW5LGkRFBQieo5Q51r1PJM,1
tz1Ks66xAkPPDp9ppEk6qLUZ71VfnDe1phM6,1
tz1LExWULm6jzNRpPCgvxANuMHN3W6rZCaRF,1
tz1LmEvSBa3aY9AG2qGiH3ijqEtbThZqcX2j,1
tz1LokKdPXHGxPuD2hKq7JPQeZy5BiV7VxGK,1
tz1LSjfgjLXmbp6VtPx7Tptxuh3Ljjbqj2Py,2
tz1LTm8Dh3MQ1qK2GPvYfMXbTux63yAn1v2z,1
tz1LtmMaNDfxE41s6sSvh1rEih9YYdpvw7hh,1
tz1LvhgKdHyJBVtb19V6fJbJC6aikrRxkYNk,2
tz1M2BxFFs3mqQecHE576Xy1mJaMAqSaNFn7,1
tz1MeRQ9XUPKAwRf5ZxgtHhJWDskKUTPqvUo,1
tz1MH5f4GBkP5oMWq55HKeqAL38pE1zSxjKr,1
tz1MpVEFoMAqAdWgakGYQz6BkcUqRM6ti9kC,1
tz1MwgPYtc8uccvYgEH4gixazzLLgayRipze,1
tz1N1vuWBsDtGMyj3No8NfeJdCBFu8v6gvSX,1
tz1NbL4iZG2AraVtBGirenGfK6m9WPsusuhh,1
tz1Nfpe3PRsRUaVhoB7mK7AmB9u78hUgz7pH,1
tz1NpUTYsVB2Hdci6ycdWdTz66GvR5oLqQp8,1
tz1NSwd8REg4bwYVgBtNFgZkFR2ch2sSF93b,1
tz1PDvJfTWagz9KnUZ8TCcbwU77WKUh6vfAY,1
tz1PeYqYxGSWHFTkoJfJ2gBe9Z35D2826QMN,1
tz1PfQDUD9UWGJ2hqHRttJU5wVtxrnPwBRa1,2
tz1PkLH6efZifS8NtKiZBWFWgDoHDmDwN9f1,1
tz1QfHrDDxQsMcvfGJ4eFWCjwMeYXGjq1cu9,1
tz1QgjmhrUD3X7kgS9mMHbUz4cS6uDiFGhAU,1
tz1QnEiLgr2AqfJEVibuS91BaosqAvrjwkFY,1
tz1QQEgtdUAR1kK6tsqHx3KdFfJKET3yADbj,1
tz1QSxDeUy6YQZj9Q9HZ5couvThGX8Qu7RzY,1
tz1QwVs1L6VafxpUofj4iR2p8xTffiTdQHNu,1
tz1RQGDzn1DTncdTsFRsDwfaC3NJG59XS7yB,1
tz1RU2acZEfJR2P6WzVKWT2vZ8vyGWnv8QKc,1
tz1RuGiZ1tEs6oLJrQfwpK3uKahn4oWGVQS9,1
tz1SbFwEWf41UGYNqAfHLoY3bJ42tWRwpR66,2
tz1SES7sUNxJgLFbbukgDoJTvQJ6zr5iXXUC,1
tz1SR5dqLZLUTWVtRMEhE2hovoos2Gb7bmPh,1
tz1SyXeYfYo8TdZ7UxDX1GN4drJGd5VtJovz,1
tz1TJeLZPhHFPfZqSFcxQrpbjU8C2SjwYnqF,1
tz1TLqerK5ewqygRUG5TcCN8tkEbuSFBQVv3,1
tz1TpU2LDii1BJNJ1t1K6N4UbNaz5gz65oZw,1
tz1TrgThszc414eMe2hdiZKL84oaFPix3C1M,1
tz1TSWEDs9wcBx2KiRzVzyzECsNpRiZaLJ1D,1
tz1TvhC3QUREF2mmuURzAFj8MurU3LehLjNW,1
tz1TVrNYYjAQ9WmRwZY4gN5WxZn2evf4ZdK9,1
tz1U2kce8BqrefzSiY2EFcjtW5EQbprmxaGA,1
tz1UC4wHex3zXF6k3VDZKT4LKDSbfiScK8Ys,1
tz1UcXyKMh5CB561T7KVVW5HsSfrRGj1DQmT,1
tz1URh4MsE1Ld6uvM5ADTbB5n2kXxJwx1BGx,1
tz1UW659VKUbjfuFaDFZYxgzdfCXKRgPw3K7,1
tz1V54mT9Y49A19983dZpykNQGaybmTzhcAB,1
tz1VQVTRYijbXCZQ9dukcRhHbVFYepKZwhAR,1
tz1VYfWeGazFmBjtAqcfEnRRs5qp8R8YviS6,1
tz1WEgnR7fKQM78ttbCnQvBw87QfM4Jimnkz,2
tz1WfSBV7yQgHXLwuyPYrA2wSp5dgcn9xjqX,2
tz1WhX3XRDvbJf4XY87Zh6JRstmqcjpM1Hfs,1
tz1WidGvmRvzcyQCbRw3BMLm5A5EETAXtMYM,1
tz1Wn3tTUZAXucbgXq1uE6jumwyfjkDLG3jr,1
tz1WqXSMDZENwSrxuVRgAnW1AEa7UCkpkwSF,1
tz1WruwwqkBCo4ftESzfYBThX2GmGGNXzPYo,1
tz1X8VRevRRkiaTbEXpCVqrMUAn2CPKAhLGD,1
tz1XBCXNbnBXXqk6tfjqrXoHeMZ922vztNr3,1
tz1XBuBrXhvLqG4YXXWE6z7bZyU7Mx6F2fiu,1
tz1XeEyDyk3EKLBg128NNMpTXB4bur5YMjnD,1
tz1XjbEZZU13QPWnpPYQvSvf691TDcF7YQx4,6
tz1XKgmxm6ZjADecBmaoL7TyRPvXiwoBzok8,1
tz1XmqF5gnTjgE6Cxu5CzaT5H7vXiFfihe95,2
tz1Xt2KbwyycQLCbqgMFvGHmbGGFrmSHAab8,1
tz1XufsxDLLGMHYusQacsWyEC2yzpa4RDbBK,1
tz1Xwa4QkJhg89GWeXP3Z67gDLvYe4DzCK8J,1
tz1XZ8qGWQ4MukcDsgrNzUZ9VubmCRZaYGMX,1
tz1YCHULFrYXSnzzpWh7hqGxHAZUNUKAVFbx,1
tz1YsZZEMHjEPUQpfvzmxhCHUAqvEoZbitNo,1
tz1Yt2YLdwnomdAxi3QwUtQfsbDyBnyuKp33,1
tz1YxAN3rn85UQxevF7w78v66xHnSQQByo2j,1
tz1ZBrF4AbTyhrncrF7n7ajDMQQKEx6wbSsy,1
tz1ZfrBRYW5ELJxdi5aPxbouzDJpkug2arNw,1
tz1ZRWhMYGxFfXnXc74bS2L991q6G5nEs6nf,1
tz2D5Z26a4JWMrtqKs128VWMPv8k5iTo1C1k,1
tz2RXosjAZdS85oCrguy7Eom757144dcU1FT,1
tz2Sj1asuMEYxmgNubyaULEb8DtUsvF2BbG5,1

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment