Skip to content

Latest commit

 

History

History
160 lines (136 loc) · 5.41 KB

readme.md

File metadata and controls

160 lines (136 loc) · 5.41 KB

Generate random names for testing purposes

I have created a small project from the data that grabs names from wiktionary, there are quite some resources like:

https://github.com/ErikGartner/randy.git https://github.com/originpath/docker-openldap-rbac.git https://github.com/roninkelt/rpgx_playing.git

I'm sure you are able to find the original wiktionary scraper better than me right now.

The result can be parsed in java, bash, sql, etc.

The schema is very simple and can easily be extended:

The output from .schema

CREATE TABLE surnames (
surname TEXT not null);
CREATE TABLE males (
male TEXT not null);
CREATE TABLE females (
female TEXT not null);

The import of the different tables was also very simple:

.mode csv
.import Female_given_names.txt females 
.import Male_given_names.txt males
.import Surnames.txt surnames

A few examples:

sqlite names.db "select male from males order by random() limit 1;"
sqlite3 names.db "select female from females order by random() limit 1;"
sqlite3 names.db "select female,surname from (select female from females order by random() limit 1) join (select surname from surnames order by random() limit 1);"

A very small standalone java main class has been created to parse the file and output one random record to stdout:

import java.io.BufferedReader;
import java.io.FileReader;
import java.util.ArrayList;
import java.util.Random;
import java.util.List;
import java.util.Arrays;
public class Names
{
	public static void main(String[] args)
	{
		try
		{
			List<String> data = new ArrayList<>();//list to store data
			String file = "Surnames.txt";//default file path
                        if (args.length != 0) file = args[0]; //overwrite with argument
			FileReader fr = new FileReader(file);
			BufferedReader br = new BufferedReader(fr);
			
			//Reading until we run out of lines
			String line = br.readLine();
			while(line != null)
			{
				data.add(line);
				line = br.readLine();
			}
			br.close();
			int index = new Random().nextInt(data.size());	
			String name = data.get(index);
			System.out.println(name);
		}
		catch(Exception e)
		{
			System.out.print(e);
		}
	}
}

Also, the compilation is simple and has been done in a script together with an example of how you could use it:

javac Names.java
while true;do java Names Female_given_names.txt| tr '\n' ' ';java Names ; done

Since the command creates a linefeed on stdout, this is simply translated to a space. You could also translate it to a comma by replacing the last argument in the tr command.

Right now working on javascript to enable creation of random names. For this you need to install Node.js on your machine. Node.js uses npm to handle packages. Nvm is a virtual package manager that allows for different package configurations for different projects. I won't go into this here.

The names.js script uses a sqlite3 database and emits a random name to stdout. The script ssn.js adds a field ssn, which is a Norwegian SSN. It would emit the entire Norwegian population, but you can see there is a potential problem with duplicate SSNs in that script. This problem is addressed in ssndb.js, and it started with the reactive db-driver for sqlite3. This was creating massive queue to sqlite3 and callbacks were being stacked up to the database, making the script unresponsive, since so many transactions queued up to the database. We then used the synchronous better-sqlite3 driver, the performance increased by magnitudes. The node ssndb.js > ssndb.txt finished in a couple of hours, while the ssh.sh script was running for a week (with some hibernates from my machine) and was not finished when I needed to restart.

If you like to try a challenge and improve the bad performing script, you could try to copy this code (WARNING! It will eat all your memory):

const sql = require('sqlite3').verbose()
const moment = require('moment')
const math = require('math')
const generator = require('fodselsnummer-generator')
var date
var numbers
var row
const gender = [ 'm', 'f']

function ssn() {

	var db = new sql.Database('ssn.db', sql.OPEN_READWRITE, (err) => {
		if (err) {
			console.log("Getting error " + err);
			exit(1);
		}
	});

	for (i=1; i< 11538; i++) {
		date = moment(new Date(+(new Date()) - Math.floor(i*1000*24*3600))).format('YYYY-MM-DD');
		console.log(i, date);
		for ( const person of gender) {
			numbers = generator.generate(date, person);
			for (j=1; j<numbers.length; j++) {
				row = getRndDb(db, numbers[j],person);
			};
		};
	};

}

function getRndDb(db, ssn, sex){

	if (sex=='f') {
		row = db.get("select female,surname from (select female from females order by random() limit 1) join (select surname from surnames order by random() limit 1);", (err, row) => { 
			console.log(ssn + ',' + sex + ',' + row.female + ',' + row.surname)
		})
	} else {
		row = db.get("select male,surname from (select male from males order by random() limit 1) join (select surname from surnames order by random() limit 1);", (err, row) => { 
			console.log(ssn + ',' + sex + ',' + row.male + ',' + row.surname)
		})
	}

}

ssn();

The femaleNames.js file can be recreated by the command:

./node_modules/csvtojson/bin/csvtojson --headers='["femaleName"]' Female_given_names.txt | jq

I hope you find this project useful an if you do I would like to get some feedback on how you use this and also if you have extended this, I would like to get a PR for a richer experience.