d56729de42
Captures client IP at job-search link-click time and threads it through to the session. Both Email and ClientIpAddress are copied from session to each result row during processing. Closes #47 Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
137 lines
5.0 KiB
C#
137 lines
5.0 KiB
C#
using System.Text.Json;
|
|
using Api.Services.Contracts;
|
|
using CvMatcher.Models.Responses;
|
|
using CvSearch.Data;
|
|
using CvSearch.Data.Entities;
|
|
using CvMatcher.Models.Settings;
|
|
using Microsoft.EntityFrameworkCore;
|
|
using Microsoft.Extensions.Options;
|
|
|
|
namespace Api.Services;
|
|
|
|
/// <summary>
|
|
/// Creates and validates one-time job search tokens, and creates the corresponding search sessions.
|
|
/// Provider configuration is read from <c>cvSearch.JobProviders</c> at session-creation time and
|
|
/// snapshotted into <c>JobSearchSessionEntity.ProviderConfigJson</c> so subsequent config changes
|
|
/// do not affect already-queued sessions.
|
|
/// Keywords are extracted by the LLM during the CV-to-job match call and stored on the token,
|
|
/// then copied to the session when the user clicks the link — no extra RAG call needed.
|
|
/// </summary>
|
|
public sealed class JobTokenService : IJobTokenService
|
|
{
|
|
private readonly CvSearchDbContext _db;
|
|
private readonly JobSearchSettings _settings;
|
|
private readonly ILogger<JobTokenService> _logger;
|
|
|
|
public JobTokenService(
|
|
CvSearchDbContext db,
|
|
IOptions<JobSearchSettings> settings,
|
|
ILogger<JobTokenService> logger)
|
|
{
|
|
_db = db;
|
|
_settings = settings.Value;
|
|
_logger = logger;
|
|
}
|
|
|
|
/// <inheritdoc />
|
|
public async Task<string?> CreateTokenAsync(string cvDocumentId, string email, string language, IReadOnlyList<string> keywords, string? location, CancellationToken ct)
|
|
{
|
|
var hasEnabledProviders = await _db.JobProviders.AnyAsync(p => p.Enabled, ct);
|
|
if (!hasEnabledProviders)
|
|
{
|
|
_logger.LogDebug("Job search token skipped — no enabled providers in cvSearch.JobProviders");
|
|
return null;
|
|
}
|
|
|
|
var token = new JobSearchTokenEntity
|
|
{
|
|
Id = Guid.NewGuid().ToString("N"),
|
|
CvDocumentId = cvDocumentId,
|
|
Email = email,
|
|
Language = language,
|
|
Keywords = string.Join(",", keywords),
|
|
Location = location,
|
|
ExpiresAt = DateTime.UtcNow.AddDays(_settings.TokenExpiryDays),
|
|
Used = false,
|
|
CreatedAt = DateTime.UtcNow
|
|
};
|
|
|
|
_db.JobSearchTokens.Add(token);
|
|
await _db.SaveChangesAsync(ct);
|
|
_logger.LogInformation("Job search token created. TokenId={TokenId}, CvDocumentId={CvDocumentId}, Keywords={Keywords}, Location={Location}", token.Id, cvDocumentId, token.Keywords, token.Location);
|
|
return token.Id;
|
|
}
|
|
|
|
/// <inheritdoc />
|
|
public async Task<string> TriggerStartAsync(string tokenId, string? clientIpAddress, CancellationToken ct)
|
|
{
|
|
var token = await _db.JobSearchTokens.FirstOrDefaultAsync(x => x.Id == tokenId, ct);
|
|
if (token is null) return StartJobSearchStatus.NotFound;
|
|
if (token.Used) return StartJobSearchStatus.AlreadyUsed;
|
|
if (token.ExpiresAt <= DateTime.UtcNow) return StartJobSearchStatus.Expired;
|
|
|
|
token.Used = true;
|
|
await _db.SaveChangesAsync(ct);
|
|
|
|
var keywords = token.Keywords;
|
|
|
|
var enabledProviders = await _db.JobProviders
|
|
.Where(p => p.Enabled)
|
|
.OrderBy(p => p.DisplayOrder)
|
|
.ToListAsync(ct);
|
|
|
|
var providerConfigJson = JsonSerializer.Serialize(
|
|
enabledProviders.Select(ToConfig).ToList(),
|
|
new JsonSerializerOptions(JsonSerializerDefaults.Web));
|
|
|
|
var session = new JobSearchSessionEntity
|
|
{
|
|
Id = Guid.NewGuid().ToString("N"),
|
|
TokenId = token.Id,
|
|
CvDocumentId = token.CvDocumentId,
|
|
Email = token.Email,
|
|
Language = token.Language,
|
|
Status = JobSearchStatus.Pending,
|
|
Keywords = keywords,
|
|
Location = token.Location,
|
|
ClientIpAddress = clientIpAddress,
|
|
ProviderConfigJson = providerConfigJson,
|
|
CreatedAt = DateTime.UtcNow
|
|
};
|
|
|
|
_db.JobSearchSessions.Add(session);
|
|
await _db.SaveChangesAsync(ct);
|
|
_logger.LogInformation(
|
|
"Job search session created. SessionId={SessionId}, Keywords={Keywords}, Providers={Providers}",
|
|
session.Id, keywords, string.Join(", ", enabledProviders.Select(p => p.Name)));
|
|
|
|
return StartJobSearchStatus.Started;
|
|
}
|
|
|
|
private static JobProviderConfig ToConfig(JobProviderEntity entity)
|
|
{
|
|
List<string> keywords;
|
|
try
|
|
{
|
|
keywords = JsonSerializer.Deserialize<List<string>>(entity.InitialKeywordsJson,
|
|
new JsonSerializerOptions(JsonSerializerDefaults.Web)) ?? [];
|
|
}
|
|
catch
|
|
{
|
|
keywords = [];
|
|
}
|
|
|
|
return new JobProviderConfig
|
|
{
|
|
Name = entity.Name,
|
|
Enabled = entity.Enabled,
|
|
SearchUrlTemplate = entity.SearchUrlTemplate,
|
|
JobLinkContains = entity.JobLinkContains,
|
|
InitialKeywords = keywords,
|
|
MaxResults = entity.MaxResults,
|
|
RequireKeywordInAnchor = entity.RequireKeywordInAnchor
|
|
};
|
|
}
|
|
|
|
}
|