Show HN:我写了一个C程序,可以为文本文件生成个性化的SHA-1哈希值
Show HN: I made a C program to create a vanity SHA-1 hash for a text file

原始链接: https://gist.github.com/o0101/77eb378b5076fe47c3336583330ac615

这段C程序尝试找到“ORIGINAL_TEXT”的修改版本,使其SHA-1哈希值以“20250327”开头。它通过切换文本中某些单词的大小写来实现这一目标。以“http”开头或包含非字母字符的单词不会被修改。 程序解析文本,识别可修改的单词,并计算可能的组合总数(最多2^32)。然后,它遍历这些组合,根据当前组合的二进制表示来翻转每个可变单词的大小写。对于每个组合,它都会生成修改后文本的SHA-1哈希值。如果哈希值与目标前缀匹配,程序会将修改后的文本保存到名为“input.txt.vanity”的文件中并退出。程序会定期通过控制台报告进度,显示尝试次数和当前哈希值前缀。如果遍历所有组合后仍未找到匹配项,程序会报告失败。

Keepamovin 创建了一个 C 程序,用于生成文本文件的自选 SHA-1 哈希值。该程序利用每个单词首字母的大小写作为位(原单词 = 0,标题大小写单词 = 1), essentially 使用文件作为“计数器 nonce”。通过迭代不同的首字母大小写组合,程序搜索与所需自选前缀匹配的哈希值。虽然 C 版本迭代次数限制在约 40 亿次,但使用 BigInt 的 JS 版本理论上可以无限迭代。其好处是细微的视觉变化,同时保持高熵值。例如,查找一个 8 位自选哈希值大约需要 20 亿次迭代,这与随机搜索的预期相符。Susam 注意到一个以“0573e7473”开头的 SHA-256 哈希值。
相关文章

原文
#include <stdio.h> #include <stdlib.h> #include <string.h> #include <ctype.h> #include <openssl/sha.h> #define TARGET_PREFIX "20250327" #define MAX_WORDS 256 #define MAX_TEXT 2048 #define MAX_ATTEMPTS (1ULL << 32) // 2^32 attempts (~4.3B, enough for 8-char prefix) const char *ORIGINAL_TEXT = "They're not particularly rare, but I think they're more pleasing to see.\n\n" "For instance this \"2822302\" makes me especially happy: https://github.com/BrowserBox/BrowserBox/commit/2822302387c4cb7ff71c4239da3dc5fa4c07e165\n\n" "It even extends up to 10 digits! That's not particularly rare - roughly 1% chance (10^10/16^10 i think) - but I just think they look nice.\n\n" "Are there any other people out there who are particularly pleased when they hit that? Sorta like hitting 7777 on the odometer, or whatever.\n\n" "I'm also a fan of the purely numeric identifiers Twitter/X uses (and has for ages).\n\n" "This all reminds me of that \"commit fuzzing tool\" that can make your log have whatever commit you want. Asked AI, turns out it's: https://github.com/not-an-aardvark/lucky-commit\n\n" "Lucky Commit! What a perfect name. I guess this Ask HN should be: Does anyone else like lucky commits?\n\n" "BTW the SHA has of this message starts with: 20250327"; typedef struct { char *lower; char *upper; int start; int len; } Word; void normalize_text(char *dest, const char *src, int len) { int j = 0, last_space = 0; for (int i = 0; i < len && src[i]; i++) { if (32 == src[i]) { if (!last_space && j > 0) dest[j++] = ' '; last_space = 1; } else { dest[j++] = src[i]; last_space = 0; } } dest[j] = '\0'; } void compute_hash(const char *text, char *hash_str) { unsigned char hash[SHA_DIGEST_LENGTH]; SHA1((unsigned char *)text, strlen(text), hash); for (int i = 0; i < SHA_DIGEST_LENGTH; i++) { sprintf(hash_str + (i * 2), "%02x", hash[i]); } hash_str[SHA_DIGEST_LENGTH * 2] = '\0'; } int is_in_url(int pos, const char *text) { const char *ptr = text; while (ptr < text + pos) { if (strncmp(ptr, "http", 4) == 0) { const char *start = ptr; while (*ptr && !isspace(*ptr)) ptr++; if (pos >= start - text && pos < ptr - text) return 1; } else { ptr++; } } return 0; } int parse_words(const char *text, Word *words, int *skip_indices, int *mutable_count) { int word_count = 0, i = 0, start = 0; int text_len = strlen(text); char *buf = malloc(text_len + 1); while (i <= text_len) { if (i == text_len || text[i] == 32) { if (i > start) { int len = i - start; strncpy(buf, text + start, len); buf[len] = '\0'; if (strncmp(buf, "http", 4) == 0 || !isalpha(buf[0])) { skip_indices[word_count] = 1; } else { words[*mutable_count].lower = strdup(buf); words[*mutable_count].upper = strdup(buf); words[*mutable_count].upper[0] = toupper(buf[0]); words[*mutable_count].lower[0] = tolower(buf[0]); words[*mutable_count].start = start; words[*mutable_count].len = len; (*mutable_count)++; } word_count++; } start = i + 1; } i++; } free(buf); return word_count; } int main(int argc, char *argv[]) { const char *file_path = (argc > 1) ? argv[1] : "input.txt"; printf("Processing file: %s\n", file_path); printf("Target prefix: %s\n", TARGET_PREFIX); Word words[MAX_WORDS]; int skip_indices[MAX_WORDS] = {0}; int mutable_count = 0; int total_words = parse_words(ORIGINAL_TEXT, words, skip_indices, &mutable_count); printf("Total mutable words: %d\n", mutable_count); unsigned long long total_combinations = (mutable_count > 64) ? MAX_ATTEMPTS : (1ULL << mutable_count); printf("Total combinations: %llu\n", total_combinations); char *current_text = malloc(MAX_TEXT); char hash_str[SHA_DIGEST_LENGTH * 2 + 1]; unsigned long long counter = 0; const unsigned long long report_interval = 1000000; FILE *fp = fopen(file_path, "w"); if (!fp) { perror("File open failed"); return 1; } fprintf(fp, "%s", ORIGINAL_TEXT); fclose(fp); while (counter < total_combinations) { strcpy(current_text, ORIGINAL_TEXT); int word_idx = 0; for (int i = 0; i < total_words && word_idx < mutable_count; i++) { if (!skip_indices[i]) { int bit = (counter >> word_idx) & 1; char *variant = bit ? words[word_idx].upper : words[word_idx].lower; memcpy(current_text + words[word_idx].start, variant, words[word_idx].len); word_idx++; } } normalize_text(current_text, current_text, MAX_TEXT); compute_hash(current_text, hash_str); if (strncmp(hash_str, TARGET_PREFIX, strlen(TARGET_PREFIX)) == 0) { printf("Success! Hash: %s\n", hash_str); printf("Counter: %llu (binary: ", counter); for (int i = mutable_count - 1; i >= 0; i--) printf("%d", (int)((counter >> i) & 1)); printf(")\n"); char *output_file = malloc(strlen(file_path) + 8); strcpy(output_file, file_path); strcat(output_file, ".vanity"); fp = fopen(output_file, "w"); fprintf(fp, "%s", current_text); fclose(fp); free(output_file); break; } if (counter % report_interval == 0) { printf("Progress: %llu/%llu - %.8s\n", counter, total_combinations, hash_str); } counter++; } if (counter == total_combinations) { printf("Failed after trying all %llu combinations\n", total_combinations); } for (int i = 0; i < mutable_count; i++) { free(words[i].lower); free(words[i].upper); } free(current_text); return 0; }
联系我们 contact @ memedata.com