b3e7123d74
Parser: most jobs read «توافقی» because the amount extractor only saw 6–10 digit numbers, missing the way Iranian ads actually state pay — «۱۵ تومان»، «۴۰ تا ۵۰ تومان»، «۲۰ میلیون»، «۲۰م» all mean MILLIONS of toman. Add colloquial detection (1–3 digit number + تومان/م/میلیون → ×1,000,000, lower bound of a range), guarded so it never matches dates/hours or a long literal-toman figure. Also: a stated amount now wins over «توافقی» (ads often say a number AND «… بقیه توافقی»). Backfill: BackfillPayAsync re-parses existing aggregated jobs/talent that have no salary and fills it in place (no AI, no ID/URL change) — wired into the post-ingest auto-cleanup and exposed as an admin button. Existing «توافقی» listings with a stated number get their salary; genuinely-negotiable ads stay توافقی. Also improves the baseSalary in JobPosting rich results. Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
218 lines
17 KiB
Plaintext
218 lines
17 KiB
Plaintext
@page
|
|
@model JobsMedical.Web.Pages.Admin.IndexModel
|
|
@{
|
|
ViewData["Title"] = "مدیریت — صف آگهیها";
|
|
}
|
|
|
|
<div class="page-head">
|
|
<div class="container">
|
|
<h1>پنل مدیریت — جمعآوری و صف آگهیها</h1>
|
|
<p class="muted">
|
|
آگهیهای جمعآوریشده از منابع را بررسی، ساختارمند و منتشر کن.
|
|
(@JalaliDate.ToPersianDigits(Model.QueueTotal.ToString()) در صف،
|
|
@JalaliDate.ToPersianDigits(Model.FlaggedTotal.ToString()) پرچمخورده)
|
|
· <a asp-page="/Admin/Overview">داشبورد</a>
|
|
· <a asp-page="/Admin/Users">کاربران</a>
|
|
· <a asp-page="/Admin/Facilities">مراکز</a>
|
|
· <a asp-page="/Admin/Reports">گزارشها</a>
|
|
· <a asp-page="/Admin/Broadcast">ارسال اعلان</a>
|
|
· <a asp-page="/Admin/Settings">تنظیمات</a>
|
|
</p>
|
|
</div>
|
|
</div>
|
|
|
|
<div class="container section">
|
|
@if (Model.IngestMessage is not null)
|
|
{
|
|
<div class="alert alert-success">✓ @Model.IngestMessage</div>
|
|
}
|
|
|
|
<div class="layout-2">
|
|
<aside class="card card-pad filter-card">
|
|
<h3>موتور جمعآوری</h3>
|
|
<p class="muted" style="font-size:13px;">منابع: @string.Join("، ", Model.SourceNames)</p>
|
|
<p class="muted" style="font-size:12px; margin:0 0 12px;">
|
|
فعال/غیرفعالسازی و تنظیم کانالها در <a asp-page="/Admin/Settings">تنظیمات</a>.
|
|
</p>
|
|
<form method="post">
|
|
<button type="submit" asp-page-handler="RunIngestion" class="btn btn-accent btn-block">اجرای جمعآوری اکنون</button>
|
|
</form>
|
|
<p class="muted" style="font-size:11px; margin:8px 0 0;">
|
|
موتور: واکشی ← حذف تکراری ← تجزیه ← اعتبارسنجی ← صف بررسی.
|
|
</p>
|
|
<form method="post" onsubmit="return confirm('⚠ همهی آیتمهای جمعآوریشده (کش) و همهی آگهیهای منتشرشده از جمعآوری حذف میشوند (آگهیهای ثبتشده توسط مراکز دستنخورده میمانند)، سپس همهچیز با هوش مصنوعی دوباره جمعآوری و افزوده میشود. این کار بازگشتناپذیر است. ادامه میدهی؟');">
|
|
<button type="submit" asp-page-handler="PurgeAndReingest" class="btn btn-outline btn-block" style="margin-top:8px; color:var(--danger); border-color:var(--danger);">
|
|
🔄 پاکسازی کش و جمعآوری مجدد با هوش مصنوعی
|
|
</button>
|
|
</form>
|
|
<p class="muted" style="font-size:11px; margin:6px 0 0;">
|
|
کش حذف تکراری و آگهیهای جمعآوریشده پاک و از نو با AI پردازش میشوند. (آگهیهای مراکز حذف نمیشوند.)
|
|
</p>
|
|
|
|
<form method="post" onsubmit="return confirm('آگهیهای «آماده به کار» از روی متنِ خامِ ذخیرهشده (بدون واکشی) دوباره با هوش مصنوعی پردازش میشوند — برای پاکسازی (حذف موارد تکراری، اصلاح نقش/گروه/تگ، افزودن موقعیت تقریبی). شیفت/استخدام دستنخورده میمانند (برای حفظ SEO). هیچ آیتمی از دست نمیرود. در پسزمینه اجرا میشود. ادامه؟');">
|
|
<button type="submit" asp-page-handler="ReprocessStored" class="btn btn-primary btn-block" style="margin-top:10px;">
|
|
🧹 پردازش مجددِ «آماده به کار»ها (امن برای SEO)
|
|
</button>
|
|
</form>
|
|
<p class="muted" style="font-size:11px; margin:6px 0 0;">
|
|
توصیهشده برای پاکسازیِ آمادهبهکارها: متنِ خام نگه داشته میشود و فقط با منطقِ جدید (یکنفر=یکآگهی، نقش پایه، گروه ثابت، تگ تمیز، موقعیت تقریبی) بازساخته میشوند. صفحاتِ «آماده به کار» ایندکس نمیشوند، پس آدرسِ ایندکسشدهای تغییر نمیکند؛ شیفت/استخدام بهمرور با ایمیجستِ تازه پاک میشوند.
|
|
</p>
|
|
|
|
<form method="post" onsubmit="return confirm('برای آگهیهای جمعآوریشدهٔ تهران که موقعیت روی نقشه ندارند، از روی متنِ آگهی محلهٔ تقریبی پیدا و مختصات تنظیم میشود. شناسه و آدرس صفحات تغییر نمیکند (امن برای SEO). ادامه؟');">
|
|
<button type="submit" asp-page-handler="BackfillCoords" class="btn btn-primary btn-block" style="margin-top:10px;">
|
|
📍 تکمیل موقعیتِ نقشه برای آگهیهای موجود
|
|
</button>
|
|
</form>
|
|
<p class="muted" style="font-size:11px; margin:6px 0 0;">
|
|
شیفت/استخدام/آمادهبهکارِ جمعآوریشدهای که مختصات ندارند، از روی محلهٔ ذکرشده در متنِ آگهی روی نقشه قرار میگیرند (محدودهٔ تقریبی). فقط مختصاتِ خالی پر میشود؛ موقعیتِ واقعیِ مراکز دستنخورده میماند.
|
|
</p>
|
|
|
|
<form method="post">
|
|
<button type="submit" asp-page-handler="BackfillPay" class="btn btn-primary btn-block" style="margin-top:10px;">
|
|
💰 استخراجِ حقوق برای آگهیهای «توافقی»
|
|
</button>
|
|
</form>
|
|
<p class="muted" style="font-size:11px; margin:6px 0 0;">
|
|
آگهیهایی که حقوقشان «توافقی» است ولی در متن مبلغ دارند (مثل «۴۰ تا ۵۰ تومان» = میلیون)، مبلغشان استخراج و ثبت میشود (درجا، بدون تغییر شناسه/آدرس).
|
|
</p>
|
|
|
|
<form method="post" onsubmit="return confirm('آگهیهای جمعآوریشدهٔ شیفت/استخدام که اکنون خارج از حوزهاند (خدمات منزل/نظافت، تبلیغاتی/آموزشی، اسپم) و استخدامهای تکراری «بایگانی» میشوند: از سایت پنهان میشوند ولی ردیفشان نگه داشته میشود (قابل بازگشت). آگهیهای معتبر و شناسه/آدرسشان دستنخورده میماند. ادامه؟');">
|
|
<button type="submit" asp-page-handler="PurgeInvalid" class="btn btn-outline btn-block" style="margin-top:10px; color:var(--danger); border-color:var(--danger);">
|
|
🧽 بایگانیِ درجای آگهیهای خارج از حوزه و تکراری (شیفت/استخدام)
|
|
</button>
|
|
</form>
|
|
<p class="muted" style="font-size:11px; margin:6px 0 0;">
|
|
فقط آگهیهایی که با صافیِ فعلی «خارج از حوزه» تشخیص داده میشوند (نه صرفاً ناقص) و استخدامهای تکراری بایگانی میشوند (وضعیت «بایگانی»، نه حذف). آگهیهای معتبر دستنخوردهاند، پس آدرسِ ایندکسشدهشان تغییر نمیکند؛ صفحهٔ موارد بایگانیشده ۴۱۰ Gone میدهد تا گوگل تمیز حذفشان کند.
|
|
</p>
|
|
|
|
<form method="post" onsubmit="return confirm('مراکز درمانیِ تکراری ادغام و مراکزِ بینام/نامعتبر (مثل «بیمارستان هستم» یا «از مدجابز») حذف میشوند؛ آگهیهایشان به مرکزِ معتبر یا «نامشخص» منتقل میشود. مراکزِ ثبتشده توسط کارفرما یا تأییدشده دستنخورده میمانند. ادامه؟');">
|
|
<button type="submit" asp-page-handler="CleanFacilities" class="btn btn-primary btn-block" style="margin-top:10px;">
|
|
🏥 ادغام مراکز تکراری و حذف مراکز بینام
|
|
</button>
|
|
</form>
|
|
|
|
<form method="post">
|
|
<button type="submit" asp-page-handler="RecorrectRoles" class="btn btn-primary btn-block" style="margin-top:10px;">
|
|
🩺 اصلاح نقشِ آگهیهای «پزشک عمومی» (دندانپزشک/متخصص و …)
|
|
</button>
|
|
</form>
|
|
|
|
<form method="post" onsubmit="return confirm('نقشهای تکراری/ترکیبی/غلطاملایی (مثل «پرستار کودک» سهتایی، «پرستار و بهیار»، «بیهیار») در نقشهای اصلی ادغام و حذف میشوند؛ آگهیهایشان به نقشِ معتبر منتقل میشود. ادامه؟');">
|
|
<button type="submit" asp-page-handler="MergeRoles" class="btn btn-primary btn-block" style="margin-top:10px;">
|
|
🏷️ ادغام نقشهای تکراری/ترکیبی/غلطاملایی
|
|
</button>
|
|
</form>
|
|
<p class="muted" style="font-size:11px; margin:6px 0 0;">
|
|
نقشهای هممعنا (تکراری، ترکیبی مثل «پرستار و بهیار»، یا غلطاملایی مثل «بیهیار») در یک نقشِ پایه ادغام میشوند تا فهرستِ نقشها تمیز شود. مدیریتِ دستی در <a asp-page="/Admin/Roles">نقشها</a>.
|
|
</p>
|
|
<p class="muted" style="font-size:11px; margin:6px 0 0;">
|
|
آگهیهایی که هوش مصنوعی به اشتباه «پزشک عمومی» زده ولی متنشان نقش دیگری دارد، از روی متن اصلاح میشوند (درجا، بدون تغییر شناسه/آدرس).
|
|
</p>
|
|
<p class="muted" style="font-size:11px; margin:6px 0 0;">
|
|
مراکز تکراری (با تطبیقِ فارسی) در یک رکورد ادغام و مراکزِ بدونِ نامِ واقعی به «نامشخص» منتقل میشوند. آگهیها حفظ میشوند؛ فقط مراکزِ جمعآوریشده و مدیریتنشده پاک میشوند.
|
|
</p>
|
|
|
|
<hr style="border:none; border-top:1px solid var(--line); margin:16px 0;" />
|
|
|
|
<h3>افزودن دستی</h3>
|
|
<form method="post">
|
|
<div class="filter-group">
|
|
<label>منبع</label>
|
|
<input type="text" name="SourceChannel" placeholder="مثلاً کانال شیفت تهران" />
|
|
</div>
|
|
<div class="filter-group">
|
|
<label>متن آگهی</label>
|
|
<textarea name="RawText" rows="5" placeholder="متن کپیشده را بچسبان..."></textarea>
|
|
</div>
|
|
<button type="submit" asp-page-handler="Add" class="btn btn-outline btn-block">افزودن به صف</button>
|
|
</form>
|
|
<p class="muted" style="font-size:12px; margin-bottom:0;">
|
|
منتشرشده: @JalaliDate.ToPersianDigits(Model.PublishedShifts.ToString()) شیفت،
|
|
@JalaliDate.ToPersianDigits(Model.PublishedJobs.ToString()) استخدام
|
|
</p>
|
|
</aside>
|
|
|
|
<div>
|
|
@if (Model.Runs.Count > 0)
|
|
{
|
|
<h2 style="font-size:20px; margin-top:0; display:flex; justify-content:space-between; align-items:center;">
|
|
تاریخچه جمعآوری
|
|
<a class="btn btn-outline" style="padding:5px 12px; font-size:13px;" asp-page="/Admin/Ingested">همه نتایج جمعآوری ←</a>
|
|
</h2>
|
|
<div class="card card-pad" style="margin-bottom:18px; overflow-x:auto;">
|
|
<table style="width:100%; border-collapse:collapse; font-size:13px; white-space:nowrap;">
|
|
<thead>
|
|
<tr style="text-align:start; color:var(--muted);">
|
|
<th style="padding:6px 8px;">زمان</th>
|
|
<th style="padding:6px 8px;">یافتشده</th>
|
|
<th style="padding:6px 8px;">صف</th>
|
|
<th style="padding:6px 8px;">منتشر</th>
|
|
<th style="padding:6px 8px;">پرچم</th>
|
|
<th style="padding:6px 8px;">اسپم</th>
|
|
<th style="padding:6px 8px;">تکراری</th>
|
|
</tr>
|
|
</thead>
|
|
<tbody>
|
|
@foreach (var run in Model.Runs)
|
|
{
|
|
<tr style="border-top:1px solid var(--line);" title="@run.Detail">
|
|
<td style="padding:6px 8px;">@JalaliDate.DateTimeLabel(run.RunAt)</td>
|
|
<td style="padding:6px 8px;">@JalaliDate.ToPersianDigits(run.Fetched.ToString())</td>
|
|
<td style="padding:6px 8px;">@JalaliDate.ToPersianDigits(run.Queued.ToString())</td>
|
|
<td style="padding:6px 8px; color:var(--primary-dark); font-weight:700;">@JalaliDate.ToPersianDigits(run.Published.ToString())</td>
|
|
<td style="padding:6px 8px;">@JalaliDate.ToPersianDigits(run.Flagged.ToString())</td>
|
|
<td style="padding:6px 8px;">@JalaliDate.ToPersianDigits(run.Spam.ToString())</td>
|
|
<td style="padding:6px 8px;">@JalaliDate.ToPersianDigits(run.Duplicates.ToString())</td>
|
|
</tr>
|
|
}
|
|
</tbody>
|
|
</table>
|
|
<p class="muted" style="font-size:11px; margin:8px 0 0;">جزئیات هر منبع را با نگهداشتن نشانگر روی هر ردیف ببین. لاگ کامل: <code dir="ltr">docker logs hamkadr_api</code></p>
|
|
</div>
|
|
}
|
|
<h2 style="font-size:20px; margin-top:0;">صف بررسی</h2>
|
|
@if (Model.Queue.Count == 0)
|
|
{
|
|
<div class="card empty-state">صف خالی است. «اجرای جمعآوری» را بزن یا آگهی اضافه کن.</div>
|
|
}
|
|
else
|
|
{
|
|
foreach (var r in Model.Queue)
|
|
{
|
|
<partial name="_RawListingRow" model="r" />
|
|
}
|
|
@if (Model.QueuePages > 1)
|
|
{
|
|
<div class="row" style="display:flex; gap:10px; justify-content:center; align-items:center; margin-top:14px;">
|
|
@if (Model.QueuePage > 1)
|
|
{ <a class="btn btn-outline" asp-route-q="@(Model.QueuePage - 1)" asp-route-f="@Model.FlaggedPage">→ قبلی</a> }
|
|
<span class="muted">صفحه @JalaliDate.ToPersianDigits(Model.QueuePage.ToString()) از @JalaliDate.ToPersianDigits(Model.QueuePages.ToString())</span>
|
|
@if (Model.QueuePage < Model.QueuePages)
|
|
{ <a class="btn btn-outline" asp-route-q="@(Model.QueuePage + 1)" asp-route-f="@Model.FlaggedPage">بعدی ←</a> }
|
|
</div>
|
|
}
|
|
}
|
|
|
|
@if (Model.FlaggedTotal > 0)
|
|
{
|
|
<h2 style="font-size:20px; margin-top:28px;">پرچمخورده (ناقص/مشکوک)</h2>
|
|
<p class="muted" style="font-size:13px;">اعتبارسنجی اینها را کامل ندانست؛ در صورت صحت میتوانی منتشرشان کنی.</p>
|
|
foreach (var r in Model.Flagged)
|
|
{
|
|
<partial name="_RawListingRow" model="r" />
|
|
}
|
|
@if (Model.FlaggedPages > 1)
|
|
{
|
|
<div class="row" style="display:flex; gap:10px; justify-content:center; align-items:center; margin-top:14px;">
|
|
@if (Model.FlaggedPage > 1)
|
|
{ <a class="btn btn-outline" asp-route-q="@Model.QueuePage" asp-route-f="@(Model.FlaggedPage - 1)">→ قبلی</a> }
|
|
<span class="muted">صفحه @JalaliDate.ToPersianDigits(Model.FlaggedPage.ToString()) از @JalaliDate.ToPersianDigits(Model.FlaggedPages.ToString())</span>
|
|
@if (Model.FlaggedPage < Model.FlaggedPages)
|
|
{ <a class="btn btn-outline" asp-route-q="@Model.QueuePage" asp-route-f="@(Model.FlaggedPage + 1)">بعدی ←</a> }
|
|
</div>
|
|
}
|
|
}
|
|
</div>
|
|
</div>
|
|
</div>
|