@prosopo/datasets-fs 3.0.43 → 3.0.77

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (227) hide show
  1. package/.turbo/turbo-build$colon$cjs.log +7 -5
  2. package/.turbo/turbo-build$colon$tsc.log +14 -11
  3. package/.turbo/turbo-build.log +8 -6
  4. package/CHANGELOG.md +250 -0
  5. package/dist/cjs/cli/cli.cjs +4 -3
  6. package/dist/cli/cli.d.ts +1 -1
  7. package/dist/cli/cli.d.ts.map +1 -1
  8. package/dist/cli/cli.js +2 -1
  9. package/dist/cli/cli.js.map +1 -1
  10. package/dist/cli/cliCommand.d.ts +1 -1
  11. package/dist/cli/cliCommand.js +1 -1
  12. package/dist/tests/data/flat/data.json +364 -0
  13. package/dist/tests/data/flat_resized/captchas_v1.json +5155 -0
  14. package/dist/tests/data/flat_resized/captchas_v2.json +5305 -0
  15. package/dist/tests/data/flat_resized/data.json +364 -0
  16. package/dist/tests/utils.d.ts.map +1 -1
  17. package/package.json +8 -6
  18. package/src/cli/cli.ts +84 -0
  19. package/src/cli/cliCommand.ts +68 -0
  20. package/src/cli/cliCommandComposite.ts +54 -0
  21. package/src/cli.ts +50 -0
  22. package/src/commands/flatten.ts +152 -0
  23. package/src/commands/generate.ts +218 -0
  24. package/src/commands/generateV1.ts +273 -0
  25. package/src/commands/generateV2.ts +293 -0
  26. package/src/commands/get.ts +115 -0
  27. package/src/commands/labels.ts +78 -0
  28. package/src/commands/relocate.ts +109 -0
  29. package/src/commands/resize.ts +171 -0
  30. package/src/dummy.ts +52 -0
  31. package/src/index.ts +14 -0
  32. package/src/tests/data/flat/data.json +364 -0
  33. package/src/tests/data/flat/images/0x1038adbe1bc5ffb5e1f180ce9fa5f727a02a43d950ba630d0a3003c95aa67609cbce9121428095180fb0442aa52b8eb4c9af3ab4497db072919e6fe65e70682b.png +0 -0
  34. package/src/tests/data/flat/images/0x1e72b814d54da74f13a5bdc56629dd0e9c6a33ac193870b3560194cff405c0c0dee3165ef4cd41cd6251df3024ac743ef8f1548a85861bc3ba680734f9ea3269.png +0 -0
  35. package/src/tests/data/flat/images/0x1f7c611483d586630dd7cc35b2114212b2a771e709dc3e05eaa400bf34c08f59e484e4cc63658c63cee63988ea1989896f1dcd2d9c5bbdfdab4b3710017bb5fd.png +0 -0
  36. package/src/tests/data/flat/images/0x20b2c989b9e8689d903490cc5cadafdf3a86a5d5e1445f42a8f77465ae659688a9cf2e3272f6c0d1945da26f1bcfcb6d21954c18c2bde806214e629d3b9d89a8.png +0 -0
  37. package/src/tests/data/flat/images/0x2a707f216df918ace936313a986faadf45141438f0d22953d1ccae7417a1a29b40bf5940a7355f035473dd5c75ef30127d8a057c1901d4f2d2950561b78b253d.png +0 -0
  38. package/src/tests/data/flat/images/0x2abca0559027978fdd6f4459745f13a4e2d10ac4f16e68dd1b4e236f19462b892d749cac2baf2d1a714b592593176e6309d192ce5363d7f4707c4e0aa6b18abb.png +0 -0
  39. package/src/tests/data/flat/images/0x2e72dcae73dcd9347896d7c32d9674b885a0beac8ae9acc24eb23e3244a169f2d4b17a2958d24dc87be1e3ae1bfe73fe1b0748a4653a191630fb86f3b58bb4a9.png +0 -0
  40. package/src/tests/data/flat/images/0x329e1e9223cc4747ff9c6306dc3affdbcceff0e629f19503ad548ca1a82da61d3781986cbd267b98f756f9bb2c8e7a879be5085028069eb3f380d5a32be81ed4.png +0 -0
  41. package/src/tests/data/flat/images/0x33a99c18c2794a36a0d524a76733a59d10743c04246263c8afe3ace1a96f73e40f964add43ec3d6d8efa9beeaa56eda6ece598fe1e3b520047b83886d6983a83.png +0 -0
  42. package/src/tests/data/flat/images/0x3992f2d799a25f4ce8e5177fa8df1d76f5afc33cc1738677d5f660ee11c963303926f6d9cd634755f0bb57a9bd0d15073a8fbe3543f569e26a34100b705ef161.png +0 -0
  43. package/src/tests/data/flat/images/0x42b08dc541114729b85ef41bb07755c8cc9fc635055bda4e8e6d80b6ee43c3d5f702bd76ba2e6abf45f4d9bba2a070c2d3dd76205f7189539b55a95f2e928ece.jpeg +0 -0
  44. package/src/tests/data/flat/images/0x4390534b3a60fbf35f456da7a0bd7c86652dbf33e2ba739b1610d5bbe008353113ceeedb68b0672fd74ad015812f02181c4e29ea9cd937a90f21e2514e79dbea.jpeg +0 -0
  45. package/src/tests/data/flat/images/0x46f2eb852f938dfd0b879a93c8098931186ecacae49a2821e4c43fade1660030bdf03938caa278a9acc78dd1751c467c857a7689c659508933010034be01be7d.png +0 -0
  46. package/src/tests/data/flat/images/0x48bb86605fcd86ad2f471746d200bdf4124dad0db82066e4624ec57921ad743b750de21504f3979e33c7261dc8f54a6c84eab6308e5752b74096b12b6dde629d.png +0 -0
  47. package/src/tests/data/flat/images/0x494bc81803193093ac9b6c0c4a5f91dc24e85641718cc0f4c9f8778f488fb14c80b10ad03d4982adbb74f2b6157bf02d30e0a6874e53780c5657d90029331601.png +0 -0
  48. package/src/tests/data/flat/images/0x4f6de6838022ac6d4c2cea1ab2f4471041c43d6bf765f249e96ddea235e96472041a8125daaadd728a7dc218535d92524f0c3335807807783034c27e3b743718.png +0 -0
  49. package/src/tests/data/flat/images/0x50c5e0685168371ad05ca4946ca1434b34769ad26dd9bb825ee504ab77260b3002ddf7a8d24474e94cdc18d5798bae7bac868336fb7c33193f9c86f77e949fba.png +0 -0
  50. package/src/tests/data/flat/images/0x53e69925b5e8dfb60c07e5dfc44742474f5004428e6f3cd8c8a3892ff86d2211cff97d6b0d4c50930786761117b17f579ca7c2d4379d7836c2435c657950b902.png +0 -0
  51. package/src/tests/data/flat/images/0x5442346936e3ce8d1b4e78f4c60e64e79b4e92b59877055557a01639643eb933f26b3e8d588361688a2f6a0c66558f06f2615028271af1eedfa66fd90b63a91b.jpeg +0 -0
  52. package/src/tests/data/flat/images/0x54ca27391394ad7b5c98c5d9a3875e4527d33dd0a1ba72401e39991e4093837a52d7f936c619722fd7ce602cda7cbe89c01b0c3b526a3dd61bcfb60b364951d7.png +0 -0
  53. package/src/tests/data/flat/images/0x5ebabefb53c4d07cd1ec002547f452ec6053840dcc28b2e507364c5307a28cbc3d9698a24f09ed01985aa290da3acd4d86aea2f9f53753f0bcca994953e474eb.png +0 -0
  54. package/src/tests/data/flat/images/0x615c5079cd4b29a062acbb916780720c0eeb91cd82d9c5f78c9a73ddda29d8842ff85a0d99e863405861d83fd275b78bcead4f61ca8edc3cb94e7093e0a134c7.png +0 -0
  55. package/src/tests/data/flat/images/0x647083b9336b47b04310e52dc5e370f64ba2bf2f78bd66196af880acd4ea1f8a132a1f8ed25be2a796d43dc40fd151548387ab4de6823758cf774cb619819be0.png +0 -0
  56. package/src/tests/data/flat/images/0x65341641633dd9baa7bb44fab08b9edd6610db285dcdc887c46221af8623711ce2b546a79743ce7240d783f2e9008751ab82cf2f8341e321b6b887b00a13ef85.png +0 -0
  57. package/src/tests/data/flat/images/0x686b4a49ecf931afd1b9c74e5a24c907b47f81d2a95579338e3949e8948da6ad3b79e8918d65b779e8bd306e7b7502463eabfc6fe94611c04a46030403d18102.png +0 -0
  58. package/src/tests/data/flat/images/0x68b2d1f5c248de0eef95f94d9d2dac3c97c8a67f5a2516b50128ae384c7674c7c88fd4e646e9ce3a2e5514cc4b0b36f2078abc57fd1cec605642f8c5ad04e160.png +0 -0
  59. package/src/tests/data/flat/images/0x69fe4e2faf45b992867f5fcc7be8aea91b332b4769affa4c081264d3a72ea180d9fbe4aac4048cab7413f8e8167565c694e0f9d9d602b24871103c932ecb326a.jpeg +0 -0
  60. package/src/tests/data/flat/images/0x6e63ab048e72fd41aa59b1434a5b95f1eb2099438ef0a171db1226cde3cda867b99e021d8628c70bedec891ce5bf0e88c302770808f069813a73d77b1771b433.png +0 -0
  61. package/src/tests/data/flat/images/0x75b79fc445b2bd17231b4f0dc170b646b14d2f732c0236b1bbef10876923ea02ecadd6aabb08ded4f70ba6a96d8789a3131e70a0d79b2644a266f1cfb5821a88.png +0 -0
  62. package/src/tests/data/flat/images/0x76521ba228624fe9522c8e2d83a1968346e9fab7228446b9f2971064e1992decaba658c11b4bad4767ddeb16e46689270a923b4cf42744d6e1130c86993cc285.png +0 -0
  63. package/src/tests/data/flat/images/0x7e9747cdd5678279cf005a57a65064080348ba230d63a3a134db327b83c05ace94338cda19002fcba60a18d6ad621477059417ae2e4518eb37655231823fc314.png +0 -0
  64. package/src/tests/data/flat/images/0x8384c8c5bce81541dd0fb19f2f573964cf318dab654f2792af5656073c1bc22bbe6e73e81e9f6cd9244bd1f7262160854c648683f6cfe3b14f751ba716f511fb.png +0 -0
  65. package/src/tests/data/flat/images/0x860e81897a1b0204fb5edf3e2355e97990b25badfba7f5bb5632b16dd6da13592fc918a8784f7f4efd0550334479b604cb0a239dc58d83739eadc84fb879996f.png +0 -0
  66. package/src/tests/data/flat/images/0x86df2da9d3e28bc5313bc1619e2127b538ef2bb08053c36e53a2ede6ba565d67aed2c65202fb180d578f7265d5b4d1c54a09f7e52b0afd12437699a9884642b7.png +0 -0
  67. package/src/tests/data/flat/images/0x90b87fdaff3e58d32f85d941cad8ac08bf84e0b4fc5c95249e1cc9099dc4aae8271eb701ffcd7617ba886db35357f40b6eda18356c5cbf3894dd3b78d1a24b97.png +0 -0
  68. package/src/tests/data/flat/images/0x93db31c0522148040443c135d25abc6e3deb37578e4a14d6dd8c7000e0ea6d69c78840e069784ee6e9f2d3524c78c33264a3f8e93eef343942ae7e15d1ba720c.png +0 -0
  69. package/src/tests/data/flat/images/0x9e682c515b2e8aade0ddfde284ee2cbc61153807602b78476738a2a4842ffe2cc34ff481ad521146a55615877bf9e62655164499dfce41aa75416c192d06ecf6.png +0 -0
  70. package/src/tests/data/flat/images/0xab7bd3f47ed451578ada004482674ded11a4fac2297552b0e214eb1fe24cdd3355763a7d983b8b8d34197266aa5c2cb1c32382699d8ba7b98a42358c67009649.png +0 -0
  71. package/src/tests/data/flat/images/0xacaa0f9c8d9bb6660c5047db2e09e2b35e567fa193db03ba32eb62ea4d27d736d61246f1949f64de111d222abc3f84ac400af013da1115140743ce7dce571dea.png +0 -0
  72. package/src/tests/data/flat/images/0xb1842196f070d2a608eac92ff489cd825f0d379ec0d02a4a736a58aea8035224e584b59dac99f7d27326791299989bbe3cdfbb9972c16d285afb6094639e7fee.png +0 -0
  73. package/src/tests/data/flat/images/0xb313323422cb06c93b37309d097869de24ff74146c2693bb5a0df5e7d26c5d8fa502bd98f0f4905ccfd00b66299f25d0ecf18b25b9b715a1bada9c3442b2ebc0.jpeg +0 -0
  74. package/src/tests/data/flat/images/0xbc0568a00d104800cc8732cda70e154cc71fe2a890d84a593071685f63b1af89ebb74736752750c0ad385ce0e2fd4acd808d2eeb18f4c5ad0d0f1eee1d57351a.png +0 -0
  75. package/src/tests/data/flat/images/0xc12ed6e8c7b674a8e6bdd0042b9a0debc8c16973195072f833b205ca7635b9d0f2da04a225347f7234f9d23ac4e579c00817a24657e06447158ed4420da47749.png +0 -0
  76. package/src/tests/data/flat/images/0xc2594429d72436068dfa5902593658843b3908eddfed9b83f5f0ecebce181278a07f35b443832d5f65ee31f9e4624155d3c729ebbc49c2f950e01348860c12c1.png +0 -0
  77. package/src/tests/data/flat/images/0xc5c2d1a2a27b758bba3551b96153c8e8fbf8122fd2b78584b23930ce885ff980ffb35cd23829ed3b3a4c3ec9472f3949bf9faeace92addcb9fa23bf87b105e0d.png +0 -0
  78. package/src/tests/data/flat/images/0xc6c918e59541d4013e41f40937290f10d6485b8c630e0ec3d0c903c85dc30dde523270df43a74bd75e966ee02bdafce588fae7a2f73c00545a27e84e1ce1b396.png +0 -0
  79. package/src/tests/data/flat/images/0xc8c98569e856f28f25783cc70ebfe249645107ed272b0ab0f83869c7686b255041d0b413755bf58cf1e0c3eaa0b89e5b55100eaeb715ed7a83c6ae289223535a.png +0 -0
  80. package/src/tests/data/flat/images/0xd76617a835930389357f28b5d7243bd25aeb45aaaf2257e249a8827bc12222a021b0cf46bb69d408fa7770f79aa0a1cbb28b0ca4438e7fc44b4f73d3b6260a38.jpeg +0 -0
  81. package/src/tests/data/flat/images/0xd9a3fba1abef24113643624d098e580808ec3952b9eda9dc5f8f1b63afa5401849c4457eb3fb1ddb17a7aa8d24e5cb54a2daafff84426177ecafccbb865770ef.jpeg +0 -0
  82. package/src/tests/data/flat/images/0xe6e6e36125ae28553d26600090be932511ae21fb6f0029485111dec2cd9c55ed1d568536e6125debb62481075c700e12cc6a3994d1846318a420cd616f82593d.png +0 -0
  83. package/src/tests/data/flat/images/0xe88a72fb70a000b24a73bc781ecc8b0ef0e66779346067f467722438266e0177a2bbc94be1ff3e0282c7484a13869e22ae8bc184590a4b59106ebc234806b243.jpeg +0 -0
  84. package/src/tests/data/flat/images/0xea764f65041a907eba37761807cd24ca38fa16517ca4bf25a114609bcc343e48da350efbca1f676337f8e251f8b3a35be8a28a4488db9280ec79de251ffb25c1.png +0 -0
  85. package/src/tests/data/flat/images/0xebdd539eceb3ba21f5ab2f541641dcb11b7c1c2429dfddd2d880d874bbdf643949b7e976c82c0a531e2bebc1f4dadcd602dcb4d706d54527bcfe30a410c0d115.png +0 -0
  86. package/src/tests/data/flat/images/0xebf84491d7b258e71d21e16b75f29163a2d5a20e6c5176d989c0ff3ff12ac3e9203193133ce222602c95cf574983fab2bb27b9e2f7b572c870f91a3bf17e5eb7.jpeg +0 -0
  87. package/src/tests/data/flat/images/0xefe4b1ba734f887e3ce2776110eab7a0d6ae7c6d74e08194be9374c1ff1a4cddd8993dc43e5cf4b6d256dc1156f3b26ceaec9182b98b8b11cfa3c8c3bb1dc8ca.png +0 -0
  88. package/src/tests/data/flat/images/0xf154d51d2999f623fc0f375a94d4d1e77d54ce4888d7af8e766b2ea5a02eed902bc56149bf348ee1a50bc2c9ab78d2c780f55b431db6ce4b23fe3b71a32ca641.png +0 -0
  89. package/src/tests/data/flat/images/0xf91877c76deba75bc5760238f61e77806a8486d1a6b09bab13e13c08de8b2cfd74a0f22057064cca459ee64be44be4301de887a72fc1c4c31acc9a02f14a345c.png +0 -0
  90. package/src/tests/data/flat/images/0xfa48acf1439c2537d1d3268796dfec0777e605c87b5092be55cbaeb1318782c6761de8d01951c99cd28a61ebc353ab6ab18de1dc6977dfc406ac1d8649a5bcfa.png +0 -0
  91. package/src/tests/data/flat/images/0xfec746312e0fcb2631eb0486a3ed88e35ba95737aadef55aa95cdcdec9fc1a3276eb2de555bbc4f467b8c900b53581a3375390064afd9c9634a45978ebb5bd3e.png +0 -0
  92. package/src/tests/data/flat/images/0xfeda2fa5da1b8b068baf09870cd4fd3a1ba01e14fc79e759c11e2c1a8ef7d75fc5cf49a86d06c94b371f30aec5c4afd1a4f583881fd62afc40cb012cd3d50b85.png +0 -0
  93. package/src/tests/data/flat_resized/captchas_v1.json +5155 -0
  94. package/src/tests/data/flat_resized/captchas_v2.json +5305 -0
  95. package/src/tests/data/flat_resized/data.json +364 -0
  96. package/src/tests/data/flat_resized/images/0x00db19a7c62a7ed3998c596215d455077cfa6ea6a9a0d9e5e9189cbde565cf1cc11e18046e01434a67937e9574c35ac3a1ed293dc2bb606e8d419aed61edf70e.png +0 -0
  97. package/src/tests/data/flat_resized/images/0x02606f6d654cc3780b67077783d8b1d0d482b224ccd0908f9e38e55a16cf6e5eb62b42559e48d881107d00fe2fe3d0d275879d700dcde504c855576548101503.png +0 -0
  98. package/src/tests/data/flat_resized/images/0x0398b3b46e8e5ffe4206919c2f0f30b94c8867097533128b12df6ff241e2e97764acf8d8fd2955875594b11c2a41f0d3e0071843f09f67a2dbad678072db4455.png +0 -0
  99. package/src/tests/data/flat_resized/images/0x0659321c2c380fab15f2405128e2439faac7cb8555e217bcafa5bc8a354d6cfce7015203e5e61101a31dfbc637a8d5c65d85b6ee48867499895957c4db5af16a.png +0 -0
  100. package/src/tests/data/flat_resized/images/0x077d377cd794017fffc8c70c12fc056d7f03f1e24c54b56bc6c13af8a75f033b0883b64502b9afa5329819ac429318ca42f4e2cb5b446ca7a3b5a169ca5354a2.png +0 -0
  101. package/src/tests/data/flat_resized/images/0x0e3aafb28186acb2b51acd0123c55650b5019c0854e1e478ef1e61aba0033cc3b9d651e314602929a077a67fde2bfbe68c99ec89823abb381a8994aa7098e63a.png +0 -0
  102. package/src/tests/data/flat_resized/images/0x1a1b08462b094f2def0d8e2659dd1bfe3dbf25fc5b06222b45471d6a84c19f6f69ed996190e05de668bd4afe6df30c6f2b942a3047503e70d95034b155695c45.png +0 -0
  103. package/src/tests/data/flat_resized/images/0x1e553d6c4afd65c3441aa00970ca9e5919c1c9efc19ee800e45d61eb26e442e7a7dd9a80b34882a18887856b14e9df9e6dd979427bbdbb9e8090df394b0844b7.png +0 -0
  104. package/src/tests/data/flat_resized/images/0x22938433d07d39fcfbb9538233f57b537d673b0c6022d2cb1b730e42a43b317a50313def95c6e651b0ca1154510ea462a4feee3a0e71bc8f42837bd0e1737c41.png +0 -0
  105. package/src/tests/data/flat_resized/images/0x25e0a29caac87e8258350c623bba6aebf0a427f0419bde1889d3b365de8f83ad5dfd4c5e8bf7952da3e8864a689b9f0ddc730d6184e85a7b420d349c3d7b8ecd.png +0 -0
  106. package/src/tests/data/flat_resized/images/0x2670edc73900aac804f21106694f1f1d46ce843abb622f5b638e7974f3dcec96ad031000205786f2745a72f8397e63a410058b16cca850418cda76c82629375c.png +0 -0
  107. package/src/tests/data/flat_resized/images/0x367af181b8fb20198197e3d7c0b52a36c2ff3ac4dc790056a7d055b0baee38c03a9038c79fc59510764de08bbb7c14307c2d21ebcb4313599782687cb40a0646.png +0 -0
  108. package/src/tests/data/flat_resized/images/0x3d02bf1afaea86d6cc0aea9e84c6745f845faa308ceac297166fd0850a944f808b10728c2f38f67ea87962ca4a1b4f4074bbaa630caf0d91998184a32e11a77d.png +0 -0
  109. package/src/tests/data/flat_resized/images/0x3e8e0f6276b6ab68722cc3e827c0a4a9d7d1575ace6527a3c76e103c0f3a4259530d3a6e626af0b19e35a5d13505013c67553fe498b0fe0e7a4060d4003324a8.png +0 -0
  110. package/src/tests/data/flat_resized/images/0x4190ae0531224be458e3a14e13af0751311ea350636a365f9144438555c56cea075a70d2c6d0c64fb8311ff674cd2b357b0c07de388a5014129e159b9f29d31b.png +0 -0
  111. package/src/tests/data/flat_resized/images/0x42c68ccc53bc41eb731952ea2adce62be61a7b81c167f6f0b2ba4ee1d2d449d45c44eddc48f9c169902cf35ba1c8739a09848faf8c632775ca257026e4a79f8f.png +0 -0
  112. package/src/tests/data/flat_resized/images/0x441aee1e0f3c90af8f5d96eece0d476a722fe93883aff615c3c966fac27e814c451eb6fb92ad18c20ae5c798fc23b55a180db9c4d416e8cb3ea0b85cec7565f7.png +0 -0
  113. package/src/tests/data/flat_resized/images/0x46640f912f03eb60cd4afbb8d6fc14b2f0c14ab93dee203681ab997cd7e43cb58d1782b31d549a158180c7a2f9b2c500c848a2047e740d8c4ac14a320d607320.png +0 -0
  114. package/src/tests/data/flat_resized/images/0x475b502023789ec81e8653253906a75d0e4e17857117b367999268a5b0b7a8638256eca95d8cc7280a275de8c884a442abb3dd3f7b9f28f2de79df39628ba8a6.png +0 -0
  115. package/src/tests/data/flat_resized/images/0x4ce6c8a8dd7b35afebf35ee25514694c947bcb082acd370fa5cee4b650548bcf0e21a864e1a99320dea5369f1b3a6b5240bcbca420972a8f1ec06a713b900b82.png +0 -0
  116. package/src/tests/data/flat_resized/images/0x4fb674a19e4db835814fffc3872df8a661694e0deec3af347cd19e199c5539e5988c72052e79f21e0035d328d522aec983ac2379209eebd56859bdd5f400a70c.png +0 -0
  117. package/src/tests/data/flat_resized/images/0x5068adcbae2342e7208e394f57b5921df657b1e53b2696cfcb885a9298597fd7645ee1f1a0ff423be46645ca3a30d929ca59a23139ecab0e6ffecd76f3bc6558.png +0 -0
  118. package/src/tests/data/flat_resized/images/0x511c1fb6c24e2db829767c0503ed3fb5d83d2fc512bd4f53879f2a639095a05b9a104f968dae26e8fa2385c9978c55fa0a646fa07694b6403e4497ad7ad10a59.png +0 -0
  119. package/src/tests/data/flat_resized/images/0x55709825a4883482be0d647137464f012f61009149ebe8d27b2ad8445064741592795a3d02da53fc9b42e049d0e8d764ba195541267baa2788cd6985197647f8.png +0 -0
  120. package/src/tests/data/flat_resized/images/0x5cafe30f3eed6c7e6f64e34a081e50317892ac7f235c5fe865e224d8363d65b1b2d9f1f6c8b5736c0f3a932740b9b31eefd7f0ec13a16e4dfca541012adc0b2d.png +0 -0
  121. package/src/tests/data/flat_resized/images/0x69926ed2721a19f01cf8829611399412e74e5780c64f00c1610aba0a832d9e440399f13471e8cc019fcc2ec372571d1b1a583a967f21fa1cd68607cadf2f7f8c.png +0 -0
  122. package/src/tests/data/flat_resized/images/0x72b1fdf0560059de27bc796fd0ac98e083277157b2ac2ee1f5812ab4e34a781aa15b7372961dc69c4f562df5143ce356b184acd9409803ce1d443ca39d257952.png +0 -0
  123. package/src/tests/data/flat_resized/images/0x760fa7f860a1e0925b827f6aa83e52d7e76a70f55e263da07c6997b6c641987f78dc208ce97cec2eb3d68a2e653acc7d308ce4e242891573b4d50281822f5485.png +0 -0
  124. package/src/tests/data/flat_resized/images/0x77fe0772ef80332ac3cb48d5c84be6fefec0ad90607ec45c509edda2a6b63e389895693821a9ee99a2d824cf71fa9bea4458ca3de8353ae60f489fa6e3961878.png +0 -0
  125. package/src/tests/data/flat_resized/images/0x7da2f1e8dd907863f6ffd9dd3a3d24d3ac28797a3429210c24be3c83e1cc6028e040250e7c765f99a85a3edd7d0a3978b46a8bfff134120b309c494d7484e24d.png +0 -0
  126. package/src/tests/data/flat_resized/images/0x7f45ee498813ebe3d314d9071d126341ab14887d8754d5571d4858f93a07c7a8d19687a76f3f2f42dc110f573626c32f52475bc6e0d8b714c2773a13cf08f8bd.png +0 -0
  127. package/src/tests/data/flat_resized/images/0x8343903b4ce3b63ffc4a66e805a18dc1cc136a91c8f67f15f261aec61c9ad58ad1856c84fe0ac97d1edc4b21f6b5b1842cc7b211328e460ed75bfad3dbb9ff16.png +0 -0
  128. package/src/tests/data/flat_resized/images/0x884c86c6b69bd29db658b3612f484bdf78e1420434082a68636f69e3d90e3cd76cf9718e099ff0ab16d7f27d8c5a3c09dcbd136d9ee01cd1d4ec72de672bea66.png +0 -0
  129. package/src/tests/data/flat_resized/images/0x88f06b2a1ce4f7e9b491a84919b59939a76baea88a48e88e95ebe48383cc15631d8524b7a9ec73cc4cba7568dcc1c7bddef4ff45c6992300e10111baf4a61971.png +0 -0
  130. package/src/tests/data/flat_resized/images/0x8ca8bc45e79bd59205fbd77b24e09e4c6ee437f27d02fc30b825037cfae6f2583f7b9d0b6a004c5a2ef9774ca1d3b45ffb94fdfabcffcdddfcd5838f2a489d3b.png +0 -0
  131. package/src/tests/data/flat_resized/images/0x8cf36665bbccaec073e346777f974edbbd7ec534e78426d6ae13dc39740271024527e945fef06d8dd78b1345c87e1b3f90f28385531910d3a0795f45a7030438.png +0 -0
  132. package/src/tests/data/flat_resized/images/0x9134327ca5ec7a5c6e1f6d8e8aa395c445d439dc32c1bbc84721b18d04fcee8394a559c362bcf6bb1e92e2d8edcf6d5dd73f1cf302a38dd646bbed2d27a6ce8f.png +0 -0
  133. package/src/tests/data/flat_resized/images/0x9401d4dde02ebdcdd64827653b90b3337485c8fb59796ee6dbefd3866404b47ca4663175b7acfad49eac177baeb8df7304bced6713f85d50767144dcb436c66c.png +0 -0
  134. package/src/tests/data/flat_resized/images/0x977448bb1fa3594a3d91b2a37acd6967ca42252f0062ed5253040f200230395e7924a4dcc392bf26ec6d2c7c1d3204db2c5fcedb4c9d21a2a90ee018df2e6ae6.png +0 -0
  135. package/src/tests/data/flat_resized/images/0x99da74b964a5dd03c619aa375ccf5dd69eb34fba3bf0e48f8e34ce2c9a1f4bf5fffcd1be4fcebebfdea61ab27bb8d36060950d6c19b7c8a89c7316a724db7a4c.png +0 -0
  136. package/src/tests/data/flat_resized/images/0x9b7dfbc8196a9d5b140f56b7eec0467c9d8a3e38ff7c017c33700040f6c98e2b79a0960f7a919d7f1ed7b435a49dce508b113ebda88331629ee8492ca29439af.png +0 -0
  137. package/src/tests/data/flat_resized/images/0xa6d77ccef52e07120c9295502b9dc972107f704abf6f4fd57dd2b8664db9fdc2c104e30e69e2c61cf1d88f8cdaeea204a2c332d49fd7ad1f61bbda7ea22bb77e.png +0 -0
  138. package/src/tests/data/flat_resized/images/0xa791e01f34c89768df8a82869d31e5ce813adb84a3cdc7be5ebe55503375fc89d49974a3a8987292b92423ded092e488130957f4725e93d21c14d0a91a2f81ed.png +0 -0
  139. package/src/tests/data/flat_resized/images/0xb31d7cab98df328d0f4a32b2969ec8e294cb4c2d1ab8bf46471a113b63c0d39c3ccb5cca55a8b46a2161c836a9fb1643e0d673e933cf21c8e4f4e46f0e1f0e96.png +0 -0
  140. package/src/tests/data/flat_resized/images/0xbc50b9d2fc9272318a3a51e555ab744a1205112d38e308caac67975ff367fabc636a6567072bdd3bda1368ccc78863c6ea115f91cb7e4a57349baca6d6d01e24.png +0 -0
  141. package/src/tests/data/flat_resized/images/0xc322ec6d7a03c9baade16ee7a71414edfe6839d7cbe5805f3632d158616d9b51f455d1daa39167359a189878905ec8d77f3e44022ec4481197f85a98d38ed345.png +0 -0
  142. package/src/tests/data/flat_resized/images/0xc5293e66e847653ca0361f1fd1460c573a66be57b769a8a41b8fb8b42da4417ff1a91c4ef037ca5b2a1f0b5797dc62db2090d88047a80a09e93eacc080e60bb2.png +0 -0
  143. package/src/tests/data/flat_resized/images/0xc8d934f2e7ac8429086669bb0788296844946834b5a3d2f3a44db276e06d7856c3782692d8bb9ba4122098d4027ca3642f32cdc7978f44493634b21b82c735c7.png +0 -0
  144. package/src/tests/data/flat_resized/images/0xc9473fa10f0aa11856eef9683cf99fa7f2bac77e5b0ab0f08a99c002289b0ca684bfbe7504dd3124dea9fb1f4be68caf09a4c3fa0bfa890a371ace855e649c16.png +0 -0
  145. package/src/tests/data/flat_resized/images/0xd3dd338481d82174ef911f76605f1c9ecf20660a65e23ab2130211c6533d018b021303d37423a641188cbe56a447beee86916cea9be935b1cbca88eb1c1c31fb.png +0 -0
  146. package/src/tests/data/flat_resized/images/0xd8df8225474e38f1034448afeee0bdcb2888f7ea09ef5f280317cf2c8a3be2f804b0e36fe5769c1f20de3de5b87e174b7bac19807cc5f181938344d53326df7f.png +0 -0
  147. package/src/tests/data/flat_resized/images/0xde8990cc77d25bdc35d4c258113dc8243af7dfb5ce3a1b00ac2e656ce84ea06c1d2a0d9740787eb1dbcc70226f18a3eaf0e16f627d828ad56d22ebc3ffa8ec49.png +0 -0
  148. package/src/tests/data/flat_resized/images/0xdf36bf20893c56da991a71f1244aaf032cf938334ce4645a856e96a2170508e5ba6b74948f43314eda7782e1169051e4ee1b276b9b8a048e08068f4d3af3e55a.png +0 -0
  149. package/src/tests/data/flat_resized/images/0xe8198b8b0e5c51c576f50853c448467a55e33005cc185225a57849c63e84267c340d2d142d391955053ebe14b0322550480341468cf5e9fcaedbf73873e9b245.png +0 -0
  150. package/src/tests/data/flat_resized/images/0xe94dcbd223afb4dfc9fb7f04095114ce611e6acb5b34384aef88728ab85dd30dcd8581bab1b66e0c6c370e75d03d7f8ac222eacb25d211f91a805ef63099fe3c.png +0 -0
  151. package/src/tests/data/flat_resized/images/0xeeeb9ef59478b2dada8acea00d7d3de0f163de5f8beea5128808fd43e616f89312476541912e0e1b29881bd58f28713bd3ebae0fbd29e0b3fb60ff310386b808.png +0 -0
  152. package/src/tests/data/flat_resized/images/0xef422469bc366be01145782e49c5b31a6c58f7d3945291c374a4e23196686121639c2fef9e7f967b37a958fda67cce10e92fc927f6c32b0f0f0ceeaea31ae0ad.png +0 -0
  153. package/src/tests/data/flat_resized/images/0xf17c8e32e5ab57f059d1c9478d2891d9354673608e8c0826d2e488807ae869a63721bf7be34855ba555f36fc9d722e4f1f78033a4996fe3562328f93c0bd88e4.png +0 -0
  154. package/src/tests/data/flat_resized/images/0xf5fe66c66f4b33e003096b4bb2c5d0ca52638b5ae4dab7b5e5925bb56c1393feb68f2552320883ba54debf91c400be7d9e6727c020bab1d07738d269f0793bcf.png +0 -0
  155. package/src/tests/data/flat_resized/images/0xf9ce625ee02e278eff475f7920b6fa7617c7078ed61e9ec50375fd47acb0c1aaeda231a779d0b76fb22a5a4e618779a70bff135bf02ebc8b68739d1fad67134c.png +0 -0
  156. package/src/tests/data/flat_resized/labels.json +13 -0
  157. package/src/tests/data/flat_resized/relocated_data.json +364 -0
  158. package/src/tests/data/hierarchical/bird/Screenshot from 2023-10-12 16-30-52.png +0 -0
  159. package/src/tests/data/hierarchical/bird/Screenshot from 2023-10-12 16-31-26.png +0 -0
  160. package/src/tests/data/hierarchical/bird/Screenshot from 2023-10-12 16-31-40.png +0 -0
  161. package/src/tests/data/hierarchical/bird/Screenshot from 2023-10-12 16-32-07.png +0 -0
  162. package/src/tests/data/hierarchical/bird/Screenshot from 2023-10-12 16-32-25.png +0 -0
  163. package/src/tests/data/hierarchical/bird/test_image_png_15.png +0 -0
  164. package/src/tests/data/hierarchical/bus/01.02.jpeg +0 -0
  165. package/src/tests/data/hierarchical/bus/01.03.jpeg +0 -0
  166. package/src/tests/data/hierarchical/bus/01.04.jpeg +0 -0
  167. package/src/tests/data/hierarchical/bus/Screenshot from 2023-10-12 16-33-02.png +0 -0
  168. package/src/tests/data/hierarchical/bus/Screenshot from 2023-10-12 16-33-21.png +0 -0
  169. package/src/tests/data/hierarchical/bus/Screenshot from 2023-10-12 16-33-32.png +0 -0
  170. package/src/tests/data/hierarchical/car/Screenshot from 2023-10-12 16-34-03.png +0 -0
  171. package/src/tests/data/hierarchical/car/Screenshot from 2023-10-12 16-34-14.png +0 -0
  172. package/src/tests/data/hierarchical/car/Screenshot from 2023-10-12 16-34-24.png +0 -0
  173. package/src/tests/data/hierarchical/car/test_image_png_25.png +0 -0
  174. package/src/tests/data/hierarchical/car/test_image_png_71.png +0 -0
  175. package/src/tests/data/hierarchical/car/test_image_png_89.png +0 -0
  176. package/src/tests/data/hierarchical/cat/test_image_png_22.png +0 -0
  177. package/src/tests/data/hierarchical/cat/test_image_png_24.png +0 -0
  178. package/src/tests/data/hierarchical/cat/test_image_png_33.png +0 -0
  179. package/src/tests/data/hierarchical/cat/test_image_png_5.png +0 -0
  180. package/src/tests/data/hierarchical/cat/test_image_png_78.png +0 -0
  181. package/src/tests/data/hierarchical/cat/test_image_png_93.png +0 -0
  182. package/src/tests/data/hierarchical/deer/Screenshot from 2023-10-12 16-34-51.png +0 -0
  183. package/src/tests/data/hierarchical/deer/Screenshot from 2023-10-12 16-34-57.png +0 -0
  184. package/src/tests/data/hierarchical/deer/Screenshot from 2023-10-12 16-35-03.png +0 -0
  185. package/src/tests/data/hierarchical/deer/test_image_png_17.png +0 -0
  186. package/src/tests/data/hierarchical/deer/test_image_png_52.png +0 -0
  187. package/src/tests/data/hierarchical/deer/test_image_png_70.png +0 -0
  188. package/src/tests/data/hierarchical/dog/test_image_png_16.png +0 -0
  189. package/src/tests/data/hierarchical/dog/test_image_png_27.png +0 -0
  190. package/src/tests/data/hierarchical/dog/test_image_png_28.png +0 -0
  191. package/src/tests/data/hierarchical/dog/test_image_png_40.png +0 -0
  192. package/src/tests/data/hierarchical/dog/test_image_png_51.png +0 -0
  193. package/src/tests/data/hierarchical/dog/test_image_png_79.png +0 -0
  194. package/src/tests/data/hierarchical/dog/test_image_png_90.png +0 -0
  195. package/src/tests/data/hierarchical/dog/test_image_png_95.png +0 -0
  196. package/src/tests/data/hierarchical/horse/Screenshot from 2023-10-12 16-35-31.png +0 -0
  197. package/src/tests/data/hierarchical/horse/Screenshot from 2023-10-12 16-35-37.png +0 -0
  198. package/src/tests/data/hierarchical/horse/Screenshot from 2023-10-12 16-35-48.png +0 -0
  199. package/src/tests/data/hierarchical/horse/test_image_png_20.png +0 -0
  200. package/src/tests/data/hierarchical/horse/test_image_png_26.png +0 -0
  201. package/src/tests/data/hierarchical/horse/test_image_png_32.png +0 -0
  202. package/src/tests/data/hierarchical/horse/test_image_png_44.png +0 -0
  203. package/src/tests/data/hierarchical/horse/test_image_png_94.png +0 -0
  204. package/src/tests/data/hierarchical/plane/01.05.jpeg +0 -0
  205. package/src/tests/data/hierarchical/plane/01.06.jpeg +0 -0
  206. package/src/tests/data/hierarchical/plane/01.07.jpeg +0 -0
  207. package/src/tests/data/hierarchical/plane/test_image_png_4.png +0 -0
  208. package/src/tests/data/hierarchical/plane/test_image_png_61.png +0 -0
  209. package/src/tests/data/hierarchical/plane/test_image_png_69.png +0 -0
  210. package/src/tests/data/hierarchical/plane/test_image_png_82.png +0 -0
  211. package/src/tests/data/hierarchical/plane/test_image_png_85.png +0 -0
  212. package/src/tests/data/hierarchical/train/01.01.jpeg +0 -0
  213. package/src/tests/data/hierarchical/train/01.08.jpeg +0 -0
  214. package/src/tests/data/hierarchical/train/01.09.jpeg +0 -0
  215. package/src/tests/data/hierarchical/train/Screenshot from 2023-10-12 16-36-19.png +0 -0
  216. package/src/tests/data/hierarchical/train/Screenshot from 2023-10-12 16-36-27.png +0 -0
  217. package/src/tests/data/hierarchical/train/Screenshot from 2023-10-12 16-36-43.png +0 -0
  218. package/src/tests/lodash.unit.test.ts +43 -0
  219. package/src/tests/mocked.unit.test.ts +341 -0
  220. package/src/tests/utils.ts +152 -0
  221. package/src/utils/input.ts +58 -0
  222. package/src/utils/inputOutput.ts +29 -0
  223. package/src/utils/output.ts +86 -0
  224. package/tsconfig.cjs.json +36 -0
  225. package/tsconfig.json +37 -0
  226. package/tsconfig.tsbuildinfo +1 -0
  227. package/tsconfig.types.json +9 -0
@@ -0,0 +1,293 @@
1
+ // Copyright 2021-2026 Prosopo (UK) Ltd.
2
+ //
3
+ // Licensed under the Apache License, Version 2.0 (the "License");
4
+ // you may not use this file except in compliance with the License.
5
+ // You may obtain a copy of the License at
6
+ //
7
+ // http://www.apache.org/licenses/LICENSE-2.0
8
+ //
9
+ // Unless required by applicable law or agreed to in writing, software
10
+ // distributed under the License is distributed on an "AS IS" BASIS,
11
+ // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12
+ // See the License for the specific language governing permissions and
13
+ // limitations under the License.
14
+
15
+ import fs from "node:fs";
16
+ import { ProsopoDatasetError } from "@prosopo/common";
17
+ import {
18
+ CaptchaTypes,
19
+ type CaptchaWithoutId,
20
+ type Captchas,
21
+ CaptchasContainerSchema,
22
+ type Item,
23
+ type RawSolution,
24
+ } from "@prosopo/types";
25
+ import { at, get } from "@prosopo/util";
26
+ import { blake2AsHex } from "@prosopo/util-crypto";
27
+ import { lodash } from "@prosopo/util/lodash";
28
+ import bcrypt from "bcrypt";
29
+ import cliProgress from "cli-progress";
30
+ import * as z from "zod";
31
+ import { Generate, ArgsSchema as GenerateArgsSchema } from "./generate.js";
32
+
33
+ export const ArgsSchema = GenerateArgsSchema.extend({
34
+ minCorrect: z.number().optional(),
35
+ minIncorrect: z.number().optional(),
36
+ minLabelled: z.number().optional(),
37
+ maxLabelled: z.number().optional(),
38
+ count: z.number().optional(),
39
+ });
40
+ export type ArgsSchemaType = typeof ArgsSchema;
41
+ export type Args = z.infer<ArgsSchemaType>;
42
+
43
+ export class GenerateV2 extends Generate<ArgsSchemaType> {
44
+ #size = 0;
45
+ #minCorrect = 0;
46
+ #saltRounds = 10;
47
+ #allowDuplicatesLabelled = false;
48
+ #allowDuplicatesUnlabelled = false;
49
+ #minIncorrect = 0;
50
+ #minLabelled = 0;
51
+ #maxLabelled = 0;
52
+ #count = 0;
53
+ #nCorrect = 0;
54
+ #nIncorrect = 0;
55
+ #nLabelled = 0;
56
+ #nUnlabelled = 0;
57
+ #target = "";
58
+ #targets: string[] = [];
59
+ #targetItems: Item[] = [];
60
+ #notTargetItems: Item[] = [];
61
+
62
+ public override getArgSchema() {
63
+ return ArgsSchema;
64
+ }
65
+
66
+ public override getDescription(): string {
67
+ return "Generate distinct captchas producing captcha challenges comprising one or more rounds, mixing labelled and unlabelled data into a single round";
68
+ }
69
+
70
+ public override getOptions() {
71
+ return lodash().merge(super.getOptions(), {
72
+ count: {
73
+ number: true,
74
+ description: "Number of captchas to generate",
75
+ },
76
+ minCorrect: {
77
+ number: true,
78
+ description: "Minimum number of correct images in each captcha",
79
+ },
80
+ minIncorrect: {
81
+ number: true,
82
+ description: "Minimum number of incorrect images in each captcha",
83
+ },
84
+ minLabelled: {
85
+ number: true,
86
+ description: "Minimum number of labelled images in each captcha",
87
+ },
88
+ maxLabelled: {
89
+ number: true,
90
+ description: "Maximum number of labelled images in each captcha",
91
+ },
92
+ });
93
+ }
94
+
95
+ private setupTarget(i: number) {
96
+ const _ = lodash();
97
+ if (this.targets.length <= 1) {
98
+ throw new ProsopoDatasetError(
99
+ new Error("not enough different labels in labelled data"),
100
+ {
101
+ translationKey: "DATASET.NOT_ENOUGH_LABELS",
102
+ },
103
+ );
104
+ }
105
+
106
+ // uniformly sample targets
107
+ const target = at(this.targets, i % this.targets.length);
108
+ const notTargets = this.targets.filter((t) => t !== target);
109
+ // how many labelled images should be in the captcha?
110
+ const nLabelled = this.unlabelled.length
111
+ ? _.random(this.#minLabelled, this.#maxLabelled)
112
+ : this.#size;
113
+ // how many correct labelled images should be in the captcha?
114
+ const maxCorrect = nLabelled - this.#minCorrect;
115
+ const nCorrect = _.random(this.#minCorrect, maxCorrect);
116
+ const nIncorrect = nLabelled - nCorrect;
117
+ const nUnlabelled = this.#size - nLabelled;
118
+
119
+ const targetItems = get(this.labelToImages, target);
120
+ const notTargetItems: Item[] = notTargets.flatMap((notTarget) =>
121
+ get(this.labelToImages, notTarget),
122
+ );
123
+
124
+ if (this.unlabelled.length > 0 && nUnlabelled > this.unlabelled.length) {
125
+ throw new ProsopoDatasetError(new Error("not enough unlabelled data"), {
126
+ translationKey: "DATASET.NOT_ENOUGH_IMAGES",
127
+ });
128
+ }
129
+ if (nCorrect > targetItems.length) {
130
+ throw new ProsopoDatasetError(
131
+ new Error(`not enough images for target (${target})`),
132
+ {
133
+ translationKey: "DATASET.NOT_ENOUGH_IMAGES",
134
+ },
135
+ );
136
+ }
137
+ if (nIncorrect > notTargetItems.length) {
138
+ throw new ProsopoDatasetError(
139
+ new Error(`not enough non-matching images for target (${target})`),
140
+ {
141
+ translationKey: "DATASET.NOT_ENOUGH_IMAGES",
142
+ },
143
+ );
144
+ }
145
+
146
+ this.#nCorrect = nCorrect;
147
+ this.#nIncorrect = nIncorrect;
148
+ this.#nLabelled = nLabelled;
149
+ this.#nUnlabelled = nUnlabelled;
150
+ this.#target = target;
151
+ this.#targets = notTargets;
152
+ this.#targetItems = targetItems;
153
+ this.#notTargetItems = notTargetItems;
154
+ }
155
+
156
+ public override async _run(args: Args) {
157
+ await super._run(args);
158
+
159
+ const outFile: string = args.output;
160
+
161
+ // get lodash (with seeded rng)
162
+ const _ = lodash();
163
+
164
+ this.#size = args.size || 9;
165
+ this.#minCorrect = args.minCorrect || 1;
166
+ this.#saltRounds = 10;
167
+ this.#allowDuplicatesLabelled =
168
+ args.allowDuplicatesLabelled || args.allowDuplicates || false;
169
+ this.#allowDuplicatesUnlabelled =
170
+ args.allowDuplicatesUnlabelled || args.allowDuplicates || false;
171
+ this.#minIncorrect = Math.max(args.minIncorrect || 1, 1); // at least 1 incorrect image
172
+ this.#minLabelled = this.#minCorrect + this.#minIncorrect; // min incorrect + correct
173
+ this.#maxLabelled = Math.min(args.maxLabelled || this.#size, this.#size); // at least 1 labelled image
174
+ this.#count = args.count || 0;
175
+
176
+ // the captcha contains n images. Each of these images are either labelled, being correct or incorrect against the target, or unlabelled. To construct one of these captchas, we need to decide how many of the images should be labelled vs unlabelled, and then how many of the labelled images should be correct vs incorrect
177
+ // in the traditional captcha, two rounds are produced, one with labelled images and the other with unlabelled images. This gives 18 images overall, 9 labels produced.
178
+ // the parameters for generation can regulate how many labels are collected vs how much of a test the captcha posses. E.g. 18 images could have 16 unlabelled and 2 labelled, or 2 unlabelled and 16 labelled. The former is a better test of the user being human, but the latter is a better for maximising label collection.
179
+ // if we focus on a single captcha round of 9 images, we must have at least 1 labelled correct image in the captcha for it to work, otherwise it's just a labelling phase, which normally isn't a problem but if we're treating these as tests for humanity too then we need some kind of test in there. (e.g. we abolish the labelled then unlabelled pattern of the challenge rounds in favour of mixing labelled and unlabelled data, but we then run a small chance of serving two completely unlabelled rounds if we don't set the min number of labelled images to 1 per captcha round)
180
+
181
+ const bar = new cliProgress.SingleBar(
182
+ {},
183
+ cliProgress.Presets.shades_classic,
184
+ );
185
+ bar.start(this.#count, 0);
186
+
187
+ // generate n captchas
188
+ const captchas: CaptchaWithoutId[] = [];
189
+ for (let i = 0; i < this.#count; i++) {
190
+ bar.increment();
191
+ // this.logger.info(`generating captcha ${i + 1} of ${count}`)
192
+ this.setupTarget(i);
193
+
194
+ // get the correct items
195
+ const correctItems: Item[] = _.sampleSize(
196
+ this.#targetItems,
197
+ this.#nCorrect,
198
+ );
199
+
200
+ // get the incorrect items
201
+ const incorrectItems: Item[] = _.sampleSize(
202
+ this.#notTargetItems,
203
+ this.#nIncorrect,
204
+ );
205
+
206
+ // get the unlabelled items
207
+ const unlabelledItems = new Set<Item>();
208
+ if (this.unlabelled.length > 0) {
209
+ while (unlabelledItems.size < this.#size - this.#nLabelled) {
210
+ // get a random image from the unlabelled data
211
+ const image = at(
212
+ this.unlabelled,
213
+ _.random(0, this.unlabelled.length - 1),
214
+ );
215
+ unlabelledItems.add(image);
216
+ }
217
+ }
218
+
219
+ const itemsConcat: Item[] = [
220
+ ...correctItems,
221
+ ...incorrectItems,
222
+ ...unlabelledItems,
223
+ ];
224
+ let indices: number[] = [...Array(itemsConcat.length).keys()];
225
+ indices = _.shuffle(indices);
226
+ const items = indices
227
+ .map((i) => at(itemsConcat, i))
228
+ .map((item) => {
229
+ return {
230
+ data: item.data,
231
+ hash: item.hash,
232
+ type: item.type,
233
+ };
234
+ });
235
+
236
+ // the first n indices are the correct items
237
+ const solution: RawSolution[] = indices
238
+ .map((index, i) => {
239
+ return {
240
+ pre: index, // the index of the item in the items array before shuffle
241
+ post: i, // the index of the item in the shuffled array
242
+ };
243
+ })
244
+ .filter((item) => item.pre < correctItems.length) // keep all items that were in the first n slots of the original item array - these were the correct items
245
+ .map((item) => {
246
+ return item.post; // return the index in the shuffled array
247
+ });
248
+
249
+ // the unlabelled indices were after the correct and incorrect
250
+ const unlabelledIndices: RawSolution[] = indices
251
+ .map((index, i) => {
252
+ return {
253
+ pre: index, // the index of the item in the items array before shuffle
254
+ post: i, // the index of the item in the shuffled array
255
+ };
256
+ })
257
+ .filter(
258
+ (item) => item.pre >= correctItems.length + incorrectItems.length,
259
+ ) // keep all items that were in the first n slots of the original item array - these were the correct items
260
+ .map((item) => {
261
+ return item.post; // return the index in the shuffled array
262
+ });
263
+
264
+ const salt = blake2AsHex(bcrypt.genSaltSync(this.#saltRounds));
265
+ // create the captcha
266
+ const captcha: CaptchaWithoutId = {
267
+ salt,
268
+ target: this.#target,
269
+ items,
270
+ solution,
271
+ unlabelled: unlabelledIndices,
272
+ };
273
+ captchas.push(captcha);
274
+ }
275
+ bar.stop();
276
+
277
+ // write to file
278
+ const output: Captchas = {
279
+ captchas,
280
+ format: CaptchaTypes.SelectAll,
281
+ };
282
+
283
+ // verify the output
284
+ this.logger.info(() => ({ msg: "verifying data" }));
285
+ CaptchasContainerSchema.parse(output);
286
+
287
+ this.logger.info(() => ({ msg: "writing data" }));
288
+ fs.mkdirSync(args.output.split("/").slice(0, -1).join("/"), {
289
+ recursive: true,
290
+ });
291
+ fs.writeFileSync(outFile, JSON.stringify(output, null, 4));
292
+ }
293
+ }
@@ -0,0 +1,115 @@
1
+ import fs from "node:fs";
2
+ import { get } from "@prosopo/util";
3
+ import { lodash } from "@prosopo/util/lodash";
4
+ import cliProgress from "cli-progress";
5
+ // Copyright 2021-2026 Prosopo (UK) Ltd.
6
+ //
7
+ // Licensed under the Apache License, Version 2.0 (the "License");
8
+ // you may not use this file except in compliance with the License.
9
+ // You may obtain a copy of the License at
10
+ //
11
+ // http://www.apache.org/licenses/LICENSE-2.0
12
+ //
13
+ // Unless required by applicable law or agreed to in writing, software
14
+ // distributed under the License is distributed on an "AS IS" BASIS,
15
+ // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
16
+ // See the License for the specific language governing permissions and
17
+ // limitations under the License.
18
+ import * as z from "zod";
19
+ import { InputArgsSchema, InputCliCommand } from "../utils/input.js";
20
+
21
+ export const ArgsSchema = InputArgsSchema.extend({});
22
+ export type ArgsSchemaType = typeof ArgsSchema;
23
+ export type Args = z.infer<ArgsSchemaType>;
24
+ export class Get extends InputCliCommand<ArgsSchemaType> {
25
+ public override getArgSchema() {
26
+ return ArgsSchema;
27
+ }
28
+
29
+ public override getOptions() {
30
+ return lodash().merge(super.getOptions(), {
31
+ input: {
32
+ description: 'JSON file containing urls under a "data" key',
33
+ },
34
+ });
35
+ }
36
+
37
+ public override async _run(args: Args) {
38
+ await super._run(args);
39
+ const bar = new cliProgress.SingleBar(
40
+ {},
41
+ cliProgress.Presets.shades_classic,
42
+ );
43
+
44
+ const list: string[] = [];
45
+ // biome-ignore lint/suspicious/noExplicitAny: TODO fix
46
+ const traverse = async (data: any) => {
47
+ if (Array.isArray(data)) {
48
+ for (let i = 0; i < data.length; i++) {
49
+ data[i] = await traverse(data[i]);
50
+ }
51
+ } else if (data instanceof Object) {
52
+ for (const key of Object.keys(data)) {
53
+ if (key === "data") {
54
+ const value = get(data, key);
55
+ const url = z.string().parse(value);
56
+ list.push(url);
57
+ } else {
58
+ await traverse(get(data, key));
59
+ }
60
+ }
61
+ }
62
+ return data;
63
+ };
64
+
65
+ const file = args.input;
66
+
67
+ // read the map file
68
+ // biome-ignore lint/suspicious/noExplicitAny: TODO fix
69
+ const data: any = JSON.parse(fs.readFileSync(file, "utf8"));
70
+ await traverse(data);
71
+
72
+ bar.start(list.length, 0);
73
+ for (const url of list) {
74
+ bar.increment();
75
+ if (url.startsWith("http")) {
76
+ try {
77
+ const response = await fetch(url);
78
+ if (!response.ok) {
79
+ this.logger.error(() => ({
80
+ url,
81
+ status: response.status,
82
+ statusText: response.statusText,
83
+ msg: `GET ${url} failed`,
84
+ }));
85
+ } else {
86
+ this.logger.info(() => ({ msg: `GET ${url} OK` }));
87
+ }
88
+ } catch (err) {
89
+ this.logger.error(() => ({
90
+ err,
91
+ data: { url },
92
+ msg: `GET ${url} failed`,
93
+ }));
94
+ }
95
+ } else {
96
+ // resolve locally
97
+ try {
98
+ fs.readFileSync(url);
99
+ this.logger.info(() => ({ msg: `GET ${url} OK` }));
100
+ } catch (err) {
101
+ this.logger.error(() => ({
102
+ err,
103
+ data: { url },
104
+ msg: `GET ${url} failed`,
105
+ }));
106
+ }
107
+ }
108
+ }
109
+ bar.stop();
110
+ }
111
+
112
+ public override getDescription(): string {
113
+ return "Test a GET request at image URLs";
114
+ }
115
+ }
@@ -0,0 +1,78 @@
1
+ import fs from "node:fs";
2
+ import { ProsopoDatasetError } from "@prosopo/common";
3
+ import { LabelledDataSchema, type LabelledItem } from "@prosopo/types";
4
+ import { lodash } from "@prosopo/util/lodash";
5
+ // Copyright 2021-2026 Prosopo (UK) Ltd.
6
+ //
7
+ // Licensed under the Apache License, Version 2.0 (the "License");
8
+ // you may not use this file except in compliance with the License.
9
+ // You may obtain a copy of the License at
10
+ //
11
+ // http://www.apache.org/licenses/LICENSE-2.0
12
+ //
13
+ // Unless required by applicable law or agreed to in writing, software
14
+ // distributed under the License is distributed on an "AS IS" BASIS,
15
+ // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
16
+ // See the License for the specific language governing permissions and
17
+ // limitations under the License.
18
+ import type * as z from "zod";
19
+ import {
20
+ InputOutputArgsSchema,
21
+ InputOutputCliCommand,
22
+ } from "../utils/inputOutput.js";
23
+
24
+ export const ArgsSchema = InputOutputArgsSchema.extend({});
25
+ export type ArgsSchemaType = typeof ArgsSchema;
26
+ export type Args = z.infer<ArgsSchemaType>;
27
+
28
+ export class Labels extends InputOutputCliCommand<ArgsSchemaType> {
29
+ public override getArgSchema() {
30
+ return ArgsSchema;
31
+ }
32
+
33
+ public override getOptions() {
34
+ return lodash().merge(super.getOptions(), {
35
+ input: {
36
+ description: "JSON file containing labelled data",
37
+ },
38
+ output: {
39
+ description: "Where to put the JSON file containing labels",
40
+ },
41
+ });
42
+ }
43
+
44
+ public override async _run(args: Args) {
45
+ await super._run(args);
46
+
47
+ const file = args.input;
48
+ if (!fs.existsSync(file)) {
49
+ throw new ProsopoDatasetError(new Error(`file does not exist: ${file}`), {
50
+ translationKey: "FS.FILE_NOT_FOUND",
51
+ });
52
+ }
53
+
54
+ const labelled: LabelledItem[] = file
55
+ ? LabelledDataSchema.parse(JSON.parse(fs.readFileSync(file, "utf8")))
56
+ .items
57
+ : [];
58
+
59
+ const labels = new Set<string>();
60
+ for (const item of labelled) {
61
+ labels.add(item.label);
62
+ }
63
+ const labelArray = Array.from(labels);
64
+ labelArray.sort();
65
+
66
+ fs.mkdirSync(args.output.split("/").slice(0, -1).join("/"), {
67
+ recursive: true,
68
+ });
69
+ fs.writeFileSync(
70
+ args.output,
71
+ JSON.stringify({ labels: labelArray }, null, 4),
72
+ );
73
+ }
74
+
75
+ public override getDescription(): string {
76
+ return "get all labels from some data";
77
+ }
78
+ }
@@ -0,0 +1,109 @@
1
+ import fs from "node:fs";
2
+ import { get } from "@prosopo/util";
3
+ import { lodash } from "@prosopo/util/lodash";
4
+ // Copyright 2021-2026 Prosopo (UK) Ltd.
5
+ //
6
+ // Licensed under the Apache License, Version 2.0 (the "License");
7
+ // you may not use this file except in compliance with the License.
8
+ // You may obtain a copy of the License at
9
+ //
10
+ // http://www.apache.org/licenses/LICENSE-2.0
11
+ //
12
+ // Unless required by applicable law or agreed to in writing, software
13
+ // distributed under the License is distributed on an "AS IS" BASIS,
14
+ // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15
+ // See the License for the specific language governing permissions and
16
+ // limitations under the License.
17
+ import * as z from "zod";
18
+ import {
19
+ InputOutputArgsSchema,
20
+ InputOutputCliCommand,
21
+ } from "../utils/inputOutput.js";
22
+
23
+ export const ArgsSchema = InputOutputArgsSchema.extend({
24
+ from: z.string(),
25
+ to: z.string(),
26
+ });
27
+ export type ArgsSchemaType = typeof ArgsSchema;
28
+ export type Args = z.infer<ArgsSchemaType>;
29
+
30
+ export class Relocate extends InputOutputCliCommand<ArgsSchemaType> {
31
+ public override getArgSchema() {
32
+ return ArgsSchema;
33
+ }
34
+
35
+ public override getOptions() {
36
+ return lodash().merge(super.getOptions(), {
37
+ input: {
38
+ description:
39
+ "A json file containing a list of objects with (at least) a url",
40
+ },
41
+ output: {
42
+ description: "Where to write the new json file containing the new urls",
43
+ },
44
+ from: {
45
+ string: true,
46
+ demand: true,
47
+ description: "The string to replace in the urls",
48
+ },
49
+ to: {
50
+ description: "The string to substitute in the urls",
51
+ string: true,
52
+ demand: true,
53
+ },
54
+ });
55
+ }
56
+
57
+ public override async _run(args: Args) {
58
+ await super._run(args);
59
+
60
+ const replace = (data: unknown, from: string, to: string) => {
61
+ if (Array.isArray(data)) {
62
+ for (let i = 0; i < data.length; i++) {
63
+ data[i] = replace(data[i], from, to);
64
+ }
65
+ } else if (typeof data === "object") {
66
+ const obj = data as {
67
+ [key: string]: unknown;
68
+ };
69
+ for (const key of Object.keys(obj)) {
70
+ if (key === "data") {
71
+ const value = get(obj, key);
72
+ if (typeof value === "string") {
73
+ if (value.startsWith(from)) {
74
+ this.logger.debug(() => ({
75
+ msg: "replacing",
76
+ data: { value },
77
+ }));
78
+ obj[key] = to + value.slice(from.length);
79
+ this.logger.debug(() => ({
80
+ msg: "replaced",
81
+ data: { value: obj[key] },
82
+ }));
83
+ }
84
+ }
85
+ } else {
86
+ obj[key] = replace(obj[key], from, to);
87
+ }
88
+ }
89
+ }
90
+ return data;
91
+ };
92
+
93
+ const file: string = args.input;
94
+ this.logger.info(() => ({ data: args, msg: "relocate" }));
95
+ // read the file
96
+ let data = JSON.parse(fs.readFileSync(file, "utf8"));
97
+ // replace the urls by recursively traversing the data
98
+ data = replace(data, args.from, args.to);
99
+ // write the file
100
+ fs.mkdirSync(args.output.split("/").slice(0, -1).join("/"), {
101
+ recursive: true,
102
+ });
103
+ fs.writeFileSync(args.output, JSON.stringify(data, null, 4));
104
+ }
105
+
106
+ public override getDescription(): string {
107
+ return "Relocate urls in a JSON file using string substitution";
108
+ }
109
+ }