fchat-rising/learn/recommend/profile-recommendation.ts

231 lines
11 KiB
TypeScript
Raw Normal View History

2023-03-12 05:43:58 +00:00
import _ from 'lodash';
2023-03-30 00:23:54 +00:00
import Axios from 'axios';
2023-03-12 05:43:58 +00:00
import { CharacterAnalysis, Matcher } from '../matcher';
import { FurryPreference, Kink, mammalSpecies, Species } from '../matcher-types';
2023-03-30 00:23:54 +00:00
import { characterImage } from '../../chat/common';
2024-01-29 01:21:09 +00:00
import { ProfileCache } from '../profile-cache';
2023-03-12 05:43:58 +00:00
export enum ProfileRecommendationLevel {
INFO = 'info',
NOTE = 'note',
CRITICAL = 'critical'
}
export interface ProfileRecommendationUrlParams {
// TBD
}
export interface ProfileRecommendation {
code: string;
level: ProfileRecommendationLevel;
title: string;
desc: string;
2023-03-30 00:23:54 +00:00
helpUrl?: string;
2023-03-12 05:43:58 +00:00
urlParams?: ProfileRecommendationUrlParams
}
export class ProfileRecommendationAnalyzer {
protected recommendations: ProfileRecommendation[] = [];
constructor(protected readonly profile: CharacterAnalysis) {
//
}
2023-03-30 00:23:54 +00:00
protected add(code: string, level: ProfileRecommendationLevel, title: string, desc: string, helpUrl?: string, urlParams?: ProfileRecommendationUrlParams): void {
this.recommendations.push({ code, level, title, desc, helpUrl, urlParams });
2023-03-12 05:43:58 +00:00
}
2023-03-30 00:23:54 +00:00
async analyze(): Promise<ProfileRecommendation[]> {
2023-03-12 05:43:58 +00:00
this.recommendations = [];
2023-03-30 00:23:54 +00:00
await this.checkPortrait();
2024-01-29 01:21:09 +00:00
await this.checkHqPortrait();
2023-03-30 00:23:54 +00:00
2023-03-12 05:43:58 +00:00
this.checkMissingProperties();
this.checkSpeciesPreferences();
this.checkKinkCounts();
this.checkCustomKinks();
this.checkImages();
this.checkInlineImage();
this.checkDescriptionLength();
return this.recommendations;
}
2023-03-30 00:23:54 +00:00
protected async checkPortrait(): Promise<void> {
2024-01-29 01:21:09 +00:00
const portraitUrl = characterImage(this.profile.character.name);
2023-03-30 00:23:54 +00:00
2024-01-29 01:21:09 +00:00
const result = await Axios.head(portraitUrl);
2023-03-30 00:23:54 +00:00
if (_.trim(result.headers['etag'] || '', '"').trim().toLowerCase() === '639d154d-16c3') {
this.add(`ADD_AVATAR`, ProfileRecommendationLevel.CRITICAL, 'Add an avatar portrait', 'Profiles with an avatar portrait stand out in chats.', 'https://wiki.f-list.net/Guide:_Character_Profiles#Avatar');
}
2023-03-12 05:43:58 +00:00
}
2024-01-29 01:21:09 +00:00
protected async checkHqPortrait(): Promise<void> {
const profileUrl = ProfileCache.detectRisingPortraitURL(this.profile.character.description);
if (!profileUrl) {
this.add(`ADD_HQ_AVATAR`, ProfileRecommendationLevel.CRITICAL, 'Add a high-quality portrait', 'Profiles with a high-quality portraits stand out in chats with other F-Chat Rising players.', 'https://github.com/hearmeneigh/fchat-rising/wiki/High%E2%80%90Quality-Portraits');
} else if (!ProfileCache.isSafeRisingPortraitURL(profileUrl)) {
this.add(`ADD_HQ_AVATAR_SAFE_DOMAIN`, ProfileRecommendationLevel.CRITICAL, 'Unsupported high-quality portrait URL', 'High-quality portraits can only point to f-list.net, freeimages.host, e621.net, iili.io, imgur.com, or redgifs.com domains.', 'https://github.com/hearmeneigh/fchat-rising/wiki/High%E2%80%90Quality-Portraits');
}
}
2023-03-12 05:43:58 +00:00
protected checkImages(): void {
if (!this.profile.character.image_count) {
2023-03-30 00:23:54 +00:00
this.add(`ADD_IMAGE`, ProfileRecommendationLevel.CRITICAL, 'Add a profile image', 'Profiles with images are more attractive to other players.', 'https://wiki.f-list.net/Guide:_Character_Profiles#Images');
2023-03-12 05:43:58 +00:00
} else if (this.profile.character.image_count > 1 && this.profile.character.image_count < 3) {
2023-03-30 00:23:54 +00:00
this.add(`ADD_MORE_IMAGES`, ProfileRecommendationLevel.NOTE, 'Add more profile images', 'Profiles with images are more attractive try to have at least 3 images in your profile.', 'https://wiki.f-list.net/Guide:_Character_Profiles#Images');
2023-03-12 05:43:58 +00:00
}
}
protected checkInlineImage(): void {
if (_.keys(this.profile.character.inlines).length < 1) {
2023-03-30 00:23:54 +00:00
this.add(`ADD_INLINE_IMAGE`, ProfileRecommendationLevel.NOTE, 'Add an inline image', 'Profiles with inline images are more engaging to other players.', 'https://wiki.f-list.net/Frequently_Asked_Questions#How_do_I_add_an_inline_image_to_my_profile.3F');
2023-03-12 05:43:58 +00:00
}
}
protected checkDescriptionLength(): void {
const desc = this.profile.character.description.trim();
if (desc.length < 20) {
2023-03-30 00:23:54 +00:00
this.add(`ADD_DESCRIPTION`, ProfileRecommendationLevel.CRITICAL, 'Add a description', 'Profiles with descriptions are more likely to draw attention from other players.', 'https://wiki.f-list.net/Guide:_Character_Profiles#Description');
2023-03-12 05:43:58 +00:00
} else if (desc.length < 400) {
2023-03-30 00:23:54 +00:00
this.add(`EXPAND_DESCRIPTION`, ProfileRecommendationLevel.NOTE, 'Extend your description', 'Long descriptions are more attractive to other players. Try expanding your description to at least 400 characters.', 'https://wiki.f-list.net/Guide:_Character_Profiles#Description');
2023-03-12 05:43:58 +00:00
}
}
protected checkCustomKinks(): void {
const counts = _.reduce(this.profile.character.customs, (accum, kink) => {
if (kink) {
accum.total += 1;
if (kink.description) {
accum.filled += 1;
}
}
return accum;
}, { filled: 0, total: 0 });
if (counts.total === 0) {
2023-03-30 00:23:54 +00:00
this.add(`ADD_CUSTOM_KINK`, ProfileRecommendationLevel.CRITICAL, 'Add custom kinks', `Custom kinks will help your profile stand out. Try adding at least 5 custom kinks.`, 'https://wiki.f-list.net/Guide:_Character_Profiles#Custom_Kinks');
2023-03-12 05:43:58 +00:00
} else if (counts.total < 5) {
2023-03-30 00:23:54 +00:00
this.add(`ADD_MORE_CUSTOM_KINKS`, ProfileRecommendationLevel.NOTE, 'Add more custom kinks', `Players pay a lot of attention to custom kinks. Try adding at least 5 custom kinks.`, 'https://wiki.f-list.net/Guide:_Character_Profiles#Custom_Kinks');
2023-03-12 05:43:58 +00:00
}
if (counts.filled < counts.total && counts.total > 0) {
2023-03-30 00:23:54 +00:00
this.add(`ADD_MORE_CUSTOM_KINK_DESCRIPTIONS`, ProfileRecommendationLevel.NOTE, 'Add descriptions to custom kinks', `Some or all of your custom kinks are missing descriptions. Add descriptions to your custom kinks to attract more players.`, 'https://wiki.f-list.net/Guide:_Character_Profiles#Custom_Kinks');
2023-03-12 05:43:58 +00:00
}
}
protected checkKinkCounts(): void {
const kinks = Matcher.getAllStandardKinks(this.profile.character);
const counts = _.reduce(kinks, (accum, kinkLevel) => {
2023-03-12 05:43:58 +00:00
if (_.isString(kinkLevel) && kinkLevel) {
accum[kinkLevel as keyof typeof accum] += 1;
}
return accum;
}, { favorite: 0, yes: 0, maybe: 0, no: 0 });
const minCountPerType = 5;
const totalCount = counts.favorite + counts.yes + counts.maybe + counts.no;
if (totalCount < 10) {
2023-03-30 00:23:54 +00:00
this.add(`ADD_MORE_KINKS`, ProfileRecommendationLevel.CRITICAL, `Add more kinks`, `You should have at least 10 kinks for the matching algorithm to work well.`, 'https://wiki.f-list.net/Guide:_Character_Profiles#Kinks');
2023-03-12 05:43:58 +00:00
} else {
_.each(counts, (count, key) => {
if (count < minCountPerType) {
2023-03-30 00:23:54 +00:00
this.add(`ADD_MORE_KINKS_${key.toString().toUpperCase()}`, ProfileRecommendationLevel.CRITICAL, `Add more '${key}' kinks`, `You should have at least ${minCountPerType} '${key}' kinks for the matching algorithm to work well.`, 'https://wiki.f-list.net/Guide:_Character_Profiles#Kinks');
2023-03-12 05:43:58 +00:00
}
});
}
}
protected checkMissingProperties(): void {
const p = this.profile;
if (p.age === null) {
2023-03-30 00:23:54 +00:00
this.add('AGE', ProfileRecommendationLevel.CRITICAL, 'Enter age', 'Specifying the age of your character will improve your matches with other players.', 'https://wiki.f-list.net/Guide:_Character_Profiles#General_Details');
2023-03-12 05:43:58 +00:00
}
if (p.orientation === null) {
2023-03-30 00:23:54 +00:00
this.add('ORIENTATION', ProfileRecommendationLevel.CRITICAL, 'Enter sexual orientation', 'Specifying the sexual orientation of your character will improve your matches with other players.', 'https://wiki.f-list.net/Guide:_Character_Profiles#General_Details');
2023-03-12 05:43:58 +00:00
}
if (p.species === null) {
2023-03-30 00:23:54 +00:00
this.add('SPECIES', ProfileRecommendationLevel.CRITICAL, 'Enter species', 'Specifying the species of your character even if it\'s \'human\' will improve your matches with other players.', 'https://wiki.f-list.net/Guide:_Character_Profiles#General_Details');
2023-03-12 05:43:58 +00:00
}
if (p.furryPreference === null) {
2023-03-30 00:23:54 +00:00
this.add('FURRY_PREFERENCE', ProfileRecommendationLevel.CRITICAL, 'Enter furry preference', 'Specifying whether you like to play with anthro characters will improve your matches with other players.', 'https://wiki.f-list.net/Guide:_Character_Profiles#RPing_Preferences');
2023-03-12 05:43:58 +00:00
}
if (p.subDomRole === null) {
2023-03-30 00:23:54 +00:00
this.add('SUB_DOM_ROLE', ProfileRecommendationLevel.CRITICAL, 'Enter sub/dom role', 'Specifying your preferred sub/dom role will improve your matches with other players.', 'https://wiki.f-list.net/Guide:_Character_Profiles#Sexual_Details');
2023-03-12 05:43:58 +00:00
}
if (p.position === null) {
2023-03-30 00:23:54 +00:00
this.add('POSITION', ProfileRecommendationLevel.CRITICAL, 'Enter position', 'Specifying your preferred position (e.g. "top", "bottom") will improve your matches with other players.', 'https://wiki.f-list.net/Guide:_Character_Profiles#Sexual_Details');
2023-03-12 05:43:58 +00:00
}
if (p.postLengthPreference === null) {
2023-03-30 00:23:54 +00:00
this.add('POST_LENGTH', ProfileRecommendationLevel.CRITICAL, 'Enter post length preference', 'Specifying your post length preference will improve your matches with other players.', 'https://wiki.f-list.net/Guide:_Character_Profiles#RPing_Preferences');
2023-03-12 05:43:58 +00:00
}
if (p.bodyType === null) {
2023-03-30 00:23:54 +00:00
this.add('BODY_TYPE', ProfileRecommendationLevel.CRITICAL, 'Enter body type', 'Specifying your character\'s body type will improve your matches with other players.', 'https://wiki.f-list.net/Guide:_Character_Profiles#General_Details');
}
if (p.gender === null) {
this.add('GENDER', ProfileRecommendationLevel.CRITICAL, 'Enter gender', 'Specifying your character\'s gender will help matching you with other players', 'https://wiki.f-list.net/Guide:_Character_Profiles#General_Details');
2023-03-12 05:43:58 +00:00
}
}
protected checkSpeciesPreferences(): void {
const p = this.profile;
const c = this.profile.character;
if (p.furryPreference === null) {
return;
}
if (p.furryPreference === FurryPreference.FurriesOnly) {
if (Matcher.getKinkPreference(c, Kink.Humans)! > 0) {
this.add('KINK_MISMATCH_FURRIES_ONLY_HUMAN', ProfileRecommendationLevel.NOTE, 'Inconsistent kink', 'Your "furries-only" profile has a positive "humans" kink. If you are open to playing with humans, consider updating your preference from "furries only" to "furs and humans".');
}
}
if (p.furryPreference === FurryPreference.HumansOnly) {
if (Matcher.getKinkPreference(c, Kink.AnimalsFerals)! >= 0 || Matcher.getKinkPreference(c, Kink.Zoophilia)! >= 0) {
// do nothing
} else {
const likedAnthros = this.getLikedAnimals();
_.each(likedAnthros, (species) => {
this.add('KINK_MISMATCH_HUMANS_ONLY_ANTHRO', ProfileRecommendationLevel.NOTE, 'Inconsistent kink', `Your "humans-only" profile has a positive "furry" kink (${Matcher.getSpeciesName(species)}). If you are open to playing with anthros, consider updating your preference from "humans only" to "furs and humans"`);
});
}
}
if (p.furryPreference !== FurryPreference.HumansOnly) {
const likedAnthros = this.getLikedAnimals();
if (likedAnthros && !_.difference(likedAnthros, [Kink.AnthroCharacters, Kink.Mammals, Kink.Humans] as any as Species[])) {
this.add('KINK_NO_SPECIES', ProfileRecommendationLevel.NOTE, 'Add preferred species', 'Specifying which anthro species you like (e.g. "equines", or "canines") in your kinks can improve your matches.');
}
}
}
protected getLikedAnimals(): Species[] {
const c = this.profile.character;
return _.filter(mammalSpecies, (species) => Matcher.getKinkPreference(c, species)! > 0);
}
}