fchat-rising/learn/profile-cache.ts

216 lines
5.8 KiB
TypeScript
Raw Normal View History

2019-07-07 01:37:15 +00:00
import * as _ from 'lodash';
import core from '../chat/core';
2020-03-14 21:24:49 +00:00
import {Character as ComplexCharacter, CharacterGroup, Guestbook} from '../site/character_page/interfaces';
import { AsyncCache } from './async-cache';
2019-07-07 21:44:32 +00:00
import { Matcher, Score, Scoring } from './matcher';
import { PermanentIndexedStore } from './store/sql-store';
2020-03-14 21:24:49 +00:00
import { CharacterImage, SimpleCharacter } from '../interfaces';
2019-09-24 19:53:43 +00:00
export interface MetaRecord {
images: CharacterImage[] | null;
groups: CharacterGroup[] | null;
2020-03-14 21:24:49 +00:00
friends: SimpleCharacter[] | null;
guestbook: Guestbook | null;
2019-09-24 19:53:43 +00:00
lastFetched: Date | null;
}
export interface CountRecord {
groupCount: number | null;
friendCount: number | null;
guestbookCount: number | null;
lastCounted: number | null;
}
2019-07-07 01:37:15 +00:00
export interface CharacterCacheRecord {
character: ComplexCharacter;
2019-07-07 01:37:15 +00:00
lastFetched: Date;
added: Date;
matchScore: number;
2019-09-24 19:53:43 +00:00
// counts?: CountRecord;
meta?: MetaRecord;
2019-07-07 01:37:15 +00:00
}
export class ProfileCache extends AsyncCache<CharacterCacheRecord> {
protected store?: PermanentIndexedStore;
2019-07-07 21:44:32 +00:00
setStore(store: PermanentIndexedStore): void {
2019-07-07 21:44:32 +00:00
this.store = store;
}
2019-07-15 16:59:16 +00:00
getSync(name: string): CharacterCacheRecord | null {
const key = AsyncCache.nameKey(name);
if (key in this.cache) {
return this.cache[key];
}
return null;
}
async get(name: string, skipStore: boolean = false): Promise<CharacterCacheRecord | null> {
const key = AsyncCache.nameKey(name);
2019-07-07 21:44:32 +00:00
if (key in this.cache) {
return this.cache[key];
} else {
if ((!this.store) || (skipStore)) {
return null;
}
2019-07-07 21:44:32 +00:00
}
const pd = await this.store.getProfile(name);
2019-07-07 21:44:32 +00:00
if (!pd) {
return null;
}
const cacheRecord = await this.register(pd.profileData, true);
cacheRecord.lastFetched = new Date(pd.lastFetched * 1000);
cacheRecord.added = new Date(pd.firstSeen * 1000);
2019-09-24 19:53:43 +00:00
cacheRecord.meta = {
lastFetched: pd.lastMetaFetched ? new Date(pd.lastMetaFetched) : null,
groups: pd.groups,
friends: pd.friends,
images: pd.images,
guestbook: pd.guestbook
};
/* cacheRecord.counts = {
lastCounted: pd.lastCounted,
groupCount: pd.groupCount,
friendCount: pd.friendCount,
guestbookCount: pd.guestbookCount
2019-09-24 19:53:43 +00:00
}; */
return cacheRecord;
2019-07-07 21:44:32 +00:00
}
2019-09-24 19:53:43 +00:00
// async registerCount(name: string, counts: CountRecord): Promise<void> {
// const record = await this.get(name);
//
// if (!record) {
// // coward's way out
// return;
// }
//
// record.counts = counts;
//
// if (this.store) {
// await this.store.updateProfileCounts(name, counts.guestbookCount, counts.friendCount, counts.groupCount);
// }
// }
async registerMeta(name: string, meta: MetaRecord): Promise<void> {
const record = await this.get(name);
if (!record) {
// coward's way out
return;
}
2019-09-24 19:53:43 +00:00
record.meta = meta;
if (this.store) {
2019-09-24 19:53:43 +00:00
await this.store.updateProfileMeta(name, meta.images, meta.guestbook, meta.friends, meta.groups);
}
}
async register(c: ComplexCharacter, skipStore: boolean = false): Promise<CharacterCacheRecord> {
const k = AsyncCache.nameKey(c.character.name);
2019-07-07 01:37:15 +00:00
const score = ProfileCache.score(c);
2019-07-07 21:44:32 +00:00
if ((this.store) && (!skipStore)) {
await this.store.storeProfile(c);
2019-07-07 21:44:32 +00:00
}
2019-07-07 01:37:15 +00:00
if (k in this.cache) {
const rExisting = this.cache[k];
rExisting.character = c;
rExisting.lastFetched = new Date();
rExisting.matchScore = score;
return rExisting;
}
const rNew = {
character: c,
lastFetched: new Date(),
added: new Date(),
matchScore: score
};
this.cache[k] = rNew;
return rNew;
}
static score(c: ComplexCharacter): number {
2019-07-07 01:37:15 +00:00
const you = core.characters.ownProfile;
if (!you) {
return 0;
}
2019-07-07 01:37:15 +00:00
const m = Matcher.generateReport(you.character, c.character);
// let mul = Math.sign(Math.min(m.you.total, m.them.total));
// if (mul === 0)
// mul = 0.5;
// const score = Math.min(m.them.total, m.you.total); // mul * (Math.abs(m.you.total) + Math.abs(m.them.total));
2019-07-08 23:27:14 +00:00
const yourScores = _.values(m.you.scores);
const theirScores = _.values(m.them.scores);
2019-07-07 01:37:15 +00:00
const finalScore = _.reduce(
2019-07-08 23:27:14 +00:00
_.concat(yourScores, theirScores),
2019-07-07 01:37:15 +00:00
(accum: Scoring | null, score: Score) => {
if (accum === null) {
return (score.score !== Scoring.NEUTRAL) ? score.score : null;
}
return (score.score === Scoring.NEUTRAL) ? accum : Math.min(accum, score.score);
},
null
);
2019-07-08 23:27:14 +00:00
if ((finalScore !== null) && (finalScore > 0)) {
// Manage edge cases where high score may not be ideal
// Nothing to score
if ((yourScores.length === 0) || (theirScores.length === 0)) {
// can't know
return Scoring.NEUTRAL;
}
// Only neutral scores given
if (
(_.every(yourScores, (n: Scoring) => n === Scoring.NEUTRAL)) ||
(_.every(theirScores, (n: Scoring) => n === Scoring.NEUTRAL))
) {
return Scoring.NEUTRAL;
}
}
2019-07-07 01:37:15 +00:00
// console.log('Profile score', c.character.name, score, m.you.total, m.them.total,
// m.you.total + m.them.total, m.you.total * m.them.total);
return (finalScore === null) ? Scoring.NEUTRAL : finalScore;
}
}