diff --git a/chat/CharacterSearch.vue b/chat/CharacterSearch.vue
index eabe63d..743b533 100644
--- a/chat/CharacterSearch.vue
+++ b/chat/CharacterSearch.vue
@@ -28,6 +28,11 @@
+
+
+
+
+
{{results.length}} {{l('characterSearch.results')}}
@@ -66,7 +71,14 @@
import {EventBus} from './preview/event-bus';
import CharacterSearchHistory from './CharacterSearchHistory.vue';
import { Matcher } from '../learn/matcher';
- import { nonAnthroSpecies, Species, speciesMapping, speciesNames } from '../learn/matcher-types';
+ import {
+ kinkMatchScoreMap,
+ kinkMatchWeights,
+ nonAnthroSpecies,
+ Species,
+ speciesMapping,
+ speciesNames
+ } from '../learn/matcher-types';
import { CharacterCacheRecord } from '../learn/profile-cache';
import Bluebird from 'bluebird';
@@ -140,6 +152,15 @@
state = 'search';
hasReceivedResults = false;
+ debugSearchJson = JSON.stringify(
+ {
+ scoreMap: kinkMatchScoreMap,
+ weights: kinkMatchWeights
+ },
+ null,
+ 2
+ );
+
private countUpdater?: ResultCountUpdater;
data: ExtendedSearchData = {
@@ -194,6 +215,39 @@
}
+ async debugUpdateResults(): Promise {
+ if (this.state !== 'results') {
+ return;
+ }
+
+ const data = JSON.parse(this.debugSearchJson);
+
+ _.assign(kinkMatchScoreMap, data.scoreMap);
+ _.assign(kinkMatchWeights, data.weights);
+
+ core.cache.profileCache.clear();
+
+ const results = this.results;
+
+ this.results = [];
+
+ await Bluebird.delay(10);
+
+ // pre-warm cache
+ await Bluebird.mapSeries(
+ results,
+ (c) => core.cache.profileCache.get(c.character.name)
+ );
+
+ this.resultsPending = this.countPendingResults(undefined, results);
+
+ this.countUpdater?.start();
+ this.resort(results);
+
+ console.log('Done!');
+ }
+
+
@Hook('mounted')
mounted(): void {
core.connection.onMessage('ERR', (data) => {
diff --git a/chat/UserView.vue b/chat/UserView.vue
index ee67f8b..d9d6fbd 100644
--- a/chat/UserView.vue
+++ b/chat/UserView.vue
@@ -6,9 +6,10 @@
import { Component, Hook, Prop, Watch } from '@f-list/vue-ts';
import Vue from 'vue';
import {Channel, Character} from '../fchat';
-import { Matcher, Score, Scoring } from '../learn/matcher';
+import { Score, Scoring } from '../learn/matcher';
import core from './core';
import { EventBus } from './preview/event-bus';
+import { kinkMatchWeights } from '../learn/matcher-types';
export function getStatusIcon(status: Character.Status): string {
@@ -71,7 +72,7 @@ export function getStatusClasses(
const cache = core.cache.profileCache.getSync(character.name);
if (cache) {
- if ((cache.match.searchScore > Matcher.UNICORN_LEVEL) && (cache.match.matchScore === Scoring.MATCH)) {
+ if ((cache.match.searchScore > kinkMatchWeights.unicornThreshold) && (cache.match.matchScore === Scoring.MATCH)) {
matchClass = 'match-found unicorn';
matchScore = 'unicorn';
} else {
diff --git a/learn/async-cache.ts b/learn/async-cache.ts
index 440a8a4..cb270c7 100644
--- a/learn/async-cache.ts
+++ b/learn/async-cache.ts
@@ -11,4 +11,8 @@ export abstract class AsyncCache {
static nameKey(name: string): string {
return Cache.nameKey(name);
}
+
+ clear(): void {
+ this.cache = {};
+ }
}
diff --git a/learn/matcher-types.ts b/learn/matcher-types.ts
index f7b1af4..76b0d81 100644
--- a/learn/matcher-types.ts
+++ b/learn/matcher-types.ts
@@ -484,32 +484,39 @@ export interface SpeciesMappingCache {
}
+export const kinkMatchWeights = {
+ logBase: 10,
+ weakMismatchThreshold: 0.3,
+ weakMatchThreshold: 0.3,
+ unicornThreshold: 8.0
+};
+
export const kinkMatchScoreMap = {
favorite: {
- favorite: 0.55,
+ favorite: 0.5,
yes: 0.25,
maybe: -0.5,
- no: -1.0
+ no: -2
},
yes: {
- favorite: 0.20,
- yes: 0.20,
- maybe: -0.25,
+ favorite: 0.3,
+ yes: 0.2,
+ maybe: -0.15,
no: -0.5
},
maybe: {
favorite: -0.5,
- yes: -0.25,
+ yes: -0.2,
maybe: 0,
no: 0
},
no: {
- favorite: -1.0,
+ favorite: -2,
yes: -0.5,
- maybe: -0.1,
+ maybe: 0,
no: 0
}
};
diff --git a/learn/matcher.ts b/learn/matcher.ts
index 5661ab3..0499116 100644
--- a/learn/matcher.ts
+++ b/learn/matcher.ts
@@ -15,7 +15,7 @@ import {
FurryPreference,
Gender, genderKinkMapping,
Kink, KinkBucketScore, kinkComparisonExclusionGroups, kinkComparisonExclusions, kinkComparisonSwaps,
- kinkMapping, kinkMatchScoreMap,
+ kinkMapping, kinkMatchScoreMap, kinkMatchWeights,
KinkPreference, likelyHuman, mammalSpecies, nonAnthroSpecies,
Orientation,
Species, SpeciesMap, speciesMapping, SpeciesMappingCache,
@@ -180,9 +180,6 @@ export class Matcher {
readonly yourAnalysis: CharacterAnalysis;
readonly theirAnalysis: CharacterAnalysis;
- static readonly UNICORN_LEVEL = 6.0;
-
-
constructor(you: Character, them: Character, yourAnalysis?: CharacterAnalysis, theirAnalysis?: CharacterAnalysis) {
this.you = you;
this.them = them;
@@ -198,8 +195,8 @@ export class Matcher {
const youThem = new Matcher(you, them, yourAnalysis, theirAnalysis);
const themYou = new Matcher(them, you, theirAnalysis, yourAnalysis);
- const youThemMatch = youThem.match();
- const themYouMatch = themYou.match();
+ const youThemMatch = youThem.match('their');
+ const themYouMatch = themYou.match('your');
const report: MatchReport = {
_isVue: true,
@@ -240,8 +237,8 @@ export class Matcher {
const youThem = new Matcher(yourAnalysis.character, theirAnalysis.character, yourAnalysis.analysis, theirAnalysis.analysis);
const themYou = new Matcher(theirAnalysis.character, yourAnalysis.character, theirAnalysis.analysis, yourAnalysis.analysis);
- const youThemMatch = youThem.match();
- const themYouMatch = themYou.match();
+ const youThemMatch = youThem.match('their');
+ const themYouMatch = themYou.match('your');
const report: MatchReport = {
_isVue: true,
@@ -377,7 +374,7 @@ export class Matcher {
return (finalScore === null) ? Scoring.NEUTRAL : finalScore;
}
- match(): MatchResult {
+ match(pronoun: string): MatchResult {
const data: MatchResult = {
you: this.you,
them: this.them,
@@ -394,7 +391,7 @@ export class Matcher {
[TagId.FurryPreference]: this.resolveFurryPairingsScore(),
[TagId.Species]: this.resolveSpeciesScore(),
[TagId.SubDomRole]: this.resolveSubDomScore(),
- [TagId.Kinks]: this.resolveKinkScore()
+ [TagId.Kinks]: this.resolveKinkScore(pronoun)
},
info: {
@@ -570,7 +567,7 @@ export class Matcher {
}
- private resolveKinkScore(): Score {
+ private resolveKinkScore(pronoun: string): Score {
const kinkScore = this.resolveKinkBucketScore('all');
log.debug('report.score.kink', this.them.name, this.you.name, kinkScore.count, kinkScore.score, kinkScore.weighted);
@@ -580,18 +577,18 @@ export class Matcher {
}
if (kinkScore.weighted < 0) {
- if (Math.abs(kinkScore.weighted) < 0.45) {
- return new Score(Scoring.WEAK_MISMATCH, 'Challenging kinks');
+ if (Math.abs(kinkScore.weighted) < kinkMatchWeights.weakMismatchThreshold) {
+ return new Score(Scoring.WEAK_MISMATCH, `Hesitant about ${pronoun} kinks`);
}
- return new Score(Scoring.MISMATCH, 'Mismatching kinks');
+ return new Score(Scoring.MISMATCH, `Dislikes ${pronoun} kinks`);
}
- if (Math.abs(kinkScore.weighted) < 0.45) {
- return new Score(Scoring.WEAK_MATCH, 'Good kinks');
+ if (Math.abs(kinkScore.weighted) < kinkMatchWeights.weakMatchThreshold) {
+ return new Score(Scoring.WEAK_MATCH, `Likes ${pronoun} kinks`);
}
- return new Score(Scoring.MATCH, 'Great kinks');
+ return new Score(Scoring.MATCH, `Loves ${pronoun} kinks`);
}
@@ -789,6 +786,8 @@ export class Matcher {
const yourKinks = this.getAllStandardKinks(this.you);
const theirKinks = this.getAllStandardKinks(this.them);
+ // let missed = 0;
+
const result: any = _.reduce(
yourKinks,
(accum, yourKinkValue: any, yourKinkId: any) => {
@@ -816,15 +815,19 @@ export class Matcher {
};
}
+ // missed += 1;
return accum;
},
{ score: 0, count: 0 }
);
+ // const yourBucketCounts = this.countKinksByBucket(yourKinks);
+ // const theirBucketCounts = this.countKinksByBucket(theirKinks);
+
result.weighted = (result.count === 0)
? 0
: (
- (Math.log(result.count) / Math.log(5)) // log 5 base
+ (Math.log(result.count) / Math.log(kinkMatchWeights.logBase)) // log 8 base
* (result.score / result.count)
);
@@ -832,6 +835,23 @@ export class Matcher {
}
+ // private countKinksByBucket(kinks: { [key: number]: KinkChoice }): { favorite: number, yes: number, maybe: number, no: number } {
+ // return _.reduce(
+ // kinks,
+ // (accum, kinkValue) => {
+ // accum[kinkValue] += 1;
+ // return accum;
+ // },
+ // {
+ // favorite: 0,
+ // yes: 0,
+ // maybe: 0,
+ // no: 0
+ // }
+ // );
+ // }
+
+
private getAllStandardKinks(c: Character): { [key: number]: KinkChoice } {
const kinks = _.pickBy(c.kinks, _.isString);