2023-08-02 16:43:26 +03:00
|
|
|
const TableImporter = require('./TableImporter');
|
2022-10-26 19:55:08 +03:00
|
|
|
const {faker} = require('@faker-js/faker');
|
|
|
|
const {faker: americanFaker} = require('@faker-js/faker/locale/en_US');
|
|
|
|
const {blogStartDate: startTime} = require('../utils/blog-info');
|
|
|
|
const generateEvents = require('../utils/event-generator');
|
|
|
|
const {luck} = require('../utils/random');
|
2022-11-03 14:08:03 +03:00
|
|
|
const dateToDatabaseString = require('../utils/database-date');
|
2024-01-05 16:42:30 +03:00
|
|
|
const debug = require('@tryghost/debug')('MembersImporter');
|
2022-10-26 19:55:08 +03:00
|
|
|
|
|
|
|
class MembersImporter extends TableImporter {
|
2023-02-16 15:11:00 +03:00
|
|
|
static table = 'members';
|
2023-08-02 16:43:26 +03:00
|
|
|
static dependencies = [];
|
|
|
|
defaultQuantity = faker.datatype.number({
|
|
|
|
min: 7000,
|
|
|
|
max: 8000
|
|
|
|
});
|
2023-02-16 15:11:00 +03:00
|
|
|
|
2023-08-02 16:43:26 +03:00
|
|
|
constructor(knex, transaction) {
|
|
|
|
super(MembersImporter.table, knex, transaction);
|
2022-10-26 19:55:08 +03:00
|
|
|
}
|
|
|
|
|
2023-08-02 16:43:26 +03:00
|
|
|
async import(quantity = this.defaultQuantity) {
|
2024-01-05 16:42:30 +03:00
|
|
|
const generateNow = Date.now();
|
|
|
|
|
2022-10-26 19:55:08 +03:00
|
|
|
this.timestamps = generateEvents({
|
|
|
|
shape: 'ease-in',
|
|
|
|
trend: 'positive',
|
2023-08-02 16:43:26 +03:00
|
|
|
total: quantity,
|
2022-10-26 19:55:08 +03:00
|
|
|
startTime,
|
|
|
|
endTime: new Date()
|
|
|
|
}).sort();
|
2024-01-05 16:42:30 +03:00
|
|
|
debug(`${this.name} generated ${this.timestamps.length} timestamps in ${Date.now() - generateNow}ms`);
|
2023-08-02 16:43:26 +03:00
|
|
|
|
|
|
|
await super.import(quantity);
|
2022-10-26 19:55:08 +03:00
|
|
|
}
|
|
|
|
|
2023-08-02 16:43:26 +03:00
|
|
|
/**
|
|
|
|
* Add open rate data to members table
|
|
|
|
*/
|
|
|
|
async finalise() {
|
|
|
|
const emailRecipients = await this.transaction.select('id', 'member_id', 'opened_at').from('email_recipients');
|
|
|
|
|
2023-02-21 18:19:36 +03:00
|
|
|
const memberData = {};
|
|
|
|
for (const emailRecipient of emailRecipients) {
|
|
|
|
if (!(emailRecipient.member_id in memberData)) {
|
|
|
|
memberData[emailRecipient.member_id] = {
|
|
|
|
emailCount: 1,
|
|
|
|
openedCount: emailRecipient.opened_at ? 1 : 0
|
|
|
|
};
|
|
|
|
} else {
|
|
|
|
memberData[emailRecipient.member_id].emailCount += 1;
|
2023-02-22 02:48:43 +03:00
|
|
|
if (emailRecipient.opened_at) {
|
2023-02-21 18:19:36 +03:00
|
|
|
memberData[emailRecipient.member_id].openedCount += 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (const [memberId, emailInfo] of Object.entries(memberData)) {
|
|
|
|
const openRate = Math.round(100 * (emailInfo.openedCount / emailInfo.emailCount));
|
2023-08-02 16:43:26 +03:00
|
|
|
await this.transaction('members').update({
|
2023-02-21 18:19:36 +03:00
|
|
|
email_count: emailInfo.emailCount,
|
|
|
|
email_opened_count: emailInfo.openedCount,
|
|
|
|
email_open_rate: emailInfo.emailCount >= 5 ? openRate : null
|
|
|
|
}).where({id: memberId});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-10-26 19:55:08 +03:00
|
|
|
generate() {
|
2024-01-05 16:42:30 +03:00
|
|
|
const id = this.fastFakeObjectId();
|
2022-10-26 19:55:08 +03:00
|
|
|
// Use name from American locale to reflect an English-speaking audience
|
|
|
|
const name = `${americanFaker.name.firstName()} ${americanFaker.name.lastName()}`;
|
2024-01-05 16:42:30 +03:00
|
|
|
const timestamp = this.timestamps.pop();
|
2022-10-26 19:55:08 +03:00
|
|
|
|
|
|
|
return {
|
|
|
|
id,
|
|
|
|
uuid: faker.datatype.uuid(),
|
2023-11-15 19:10:28 +03:00
|
|
|
transient_id: faker.datatype.uuid(),
|
2023-12-07 14:25:49 +03:00
|
|
|
email: `${name.replace(' ', '.').replace(/[^a-zA-Z0-9]/g, '').toLowerCase()}${faker.datatype.number({min: 0, max: 999999})}@example.com`,
|
2024-01-15 18:23:49 +03:00
|
|
|
status: luck(5) ? 'comped' : luck(15) ? 'paid' : 'free',
|
2022-10-26 19:55:08 +03:00
|
|
|
name: name,
|
|
|
|
expertise: luck(30) ? faker.name.jobTitle() : undefined,
|
|
|
|
geolocation: JSON.stringify({
|
|
|
|
organization_name: faker.company.name(),
|
|
|
|
region: faker.address.state(),
|
|
|
|
accuracy: 50,
|
|
|
|
asn: parseInt(faker.random.numeric(4)),
|
|
|
|
organization: `${faker.random.alpha({count: 2, casing: 'upper'})}${faker.random.numeric(4)} ${faker.company.name()}`,
|
|
|
|
timezone: faker.address.timeZone(),
|
|
|
|
longitude: faker.address.longitude(),
|
|
|
|
country_code3: faker.address.countryCode('alpha-3'),
|
|
|
|
area_code: '0',
|
|
|
|
ip: faker.internet.ipv4(),
|
|
|
|
city: faker.address.cityName(),
|
|
|
|
country: faker.address.country(),
|
|
|
|
continent_code: 'EU',
|
|
|
|
country_code: faker.address.countryCode('alpha-2'),
|
|
|
|
latitude: faker.address.latitude()
|
|
|
|
}),
|
|
|
|
email_count: 0, // Depends on number of emails sent since created_at, the newsletter they're a part of and subscription status
|
|
|
|
email_opened_count: 0,
|
|
|
|
email_open_rate: null,
|
|
|
|
// 40% of users logged in within a week, 60% sometime since registering
|
2022-11-03 14:08:03 +03:00
|
|
|
last_seen_at: luck(40) ? dateToDatabaseString(faker.date.recent(7)) : dateToDatabaseString(faker.date.between(timestamp, new Date())),
|
|
|
|
created_at: dateToDatabaseString(timestamp),
|
2022-10-26 19:55:08 +03:00
|
|
|
created_by: id,
|
2022-11-03 14:08:03 +03:00
|
|
|
updated_at: dateToDatabaseString(timestamp)
|
2022-10-26 19:55:08 +03:00
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
module.exports = MembersImporter;
|