From 71f74dd85ac4e28196d19fb43d93f0fd9e256ae4 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=90=B4=E6=96=87=E5=B3=B0?= Date: Fri, 10 Apr 2026 14:28:04 +0800 Subject: [PATCH] =?UTF-8?q?=E4=BC=98=E5=8C=96=E4=BC=98=E5=85=88=E8=BF=9E?= =?UTF-8?q?=E6=8E=A5=E9=80=BB=E8=BE=91?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- crawler/crawler.go | 93 ++++++++++++++++--- .../{index-ClaCiNQl.js => index-CiQvUT0P.js} | 2 +- dist/index.html | 2 +- 3 files changed, 80 insertions(+), 17 deletions(-) rename dist/assets/{index-ClaCiNQl.js => index-CiQvUT0P.js} (77%) diff --git a/crawler/crawler.go b/crawler/crawler.go index 59a1ce6..4779858 100644 --- a/crawler/crawler.go +++ b/crawler/crawler.go @@ -78,6 +78,10 @@ type Crawler struct { pending int64 // 待处理的 Priority URL 数量(入队但未开始) active int64 // 正在处理的 Priority URL 数量 } + + // ---- Priority 子链接优先队列(来自 priority worker 的子链接会优先爬取)---- + priorityChildrenMu sync.Mutex + priorityChildren []string // Priority URL 产生的子链接(优先处理) } // 全局活跃线程计数器(跨包可读,无需持有 Crawler 引用) @@ -103,12 +107,26 @@ var globalPriorityStatus struct { // GlobalPriorityStatus 返回当前全局 Priority Worker 状态。 func GlobalPriorityStatus() map[string]interface{} { return map[string]interface{}{ - "pending": atomic.LoadInt64(&globalPriorityStatus.pending), - "active": atomic.LoadInt64(&globalPriorityStatus.active), - "max_workers": priorityMaxWorkers, + "pending": atomic.LoadInt64(&globalPriorityStatus.pending), + "active": atomic.LoadInt64(&globalPriorityStatus.active), + "max_workers": priorityMaxWorkers, + "children_queue": atomic.LoadInt64(&globalPriorityChildren), } } +// 全局 Priority 子链接队列长度(跨 Crawler 实例共享) +var globalPriorityChildren int64 + +// IncrementPriorityChildren 增加 priorityChildren 计数。 +func IncrementPriorityChildren(n int64) { + atomic.AddInt64(&globalPriorityChildren, n) +} + +// DecrementPriorityChildren 减少 priorityChildren 计数。 +func DecrementPriorityChildren(n int64) { + atomic.AddInt64(&globalPriorityChildren, -n) +} + // New 创建一个 Crawler 实例。 // prosperMap 由 info 模块加载,传入域名繁荣值用于调度优先级计算。 func New(db *storage.DB, a *analyzer.Analyzer, prosperMap map[string]float64) *Crawler { @@ -225,7 +243,18 @@ func (c *Crawler) runPriorityWorker() { // 直接调用 visitURL,绕过队列调度 hrefs := c.visitURL(rawURL) - // 收集的子链接正常进入 BFS 队列(由调用方处理,这里只负责爬取本身) + // 将子链接加入优先队列(保持优先级) + if len(hrefs) > 0 { + c.priorityChildrenMu.Lock() + for _, child := range hrefs { + if !c.isVisited(child) { + c.priorityChildren = append(c.priorityChildren, child) + } + } + IncrementPriorityChildren(int64(len(hrefs))) + c.priorityChildrenMu.Unlock() + } + log.Printf("[crawler] priority crawl done: %s (%d child links)", rawURL, len(hrefs)) // 清理已访问的 priority URL(防止重复爬取) @@ -312,6 +341,17 @@ func (c *Crawler) Run(entryURL string, maxEpoch int) { // 每轮 epoch 从 config 读取最新 workers 值,支持运行时动态调整 workers := config.CrawlerWorkers() + // ---- 优先处理 priorityChildren 队列(来自 priority worker 的子链接)---- + var priorityQueue []string + c.priorityChildrenMu.Lock() + if len(c.priorityChildren) > 0 { + priorityQueue = c.priorityChildren + // 更新全局计数器:这些 URL 即将被处理 + DecrementPriorityChildren(int64(len(priorityQueue))) + log.Printf("[crawler] epoch %d/%d processing %d priority children first", ep+1, maxEpoch, len(priorityQueue)) + } + c.priorityChildrenMu.Unlock() + // 每轮开始前:拉取 priority URLs,插入队列前端 priorityAdded := c.fetchAndApplyPriorityURLs(&queue) if priorityAdded > 0 { @@ -324,7 +364,7 @@ func (c *Crawler) Run(entryURL string, maxEpoch int) { c.markVisited(u) } - // 并发抓取本轮所有 URL + // ---- 并发抓取本轮所有 URL ---- var ( newLinks []URLWeight // 收集下一轮候选 URL mu sync.Mutex // 保护 newLinks 的并发写入 @@ -333,6 +373,7 @@ func (c *Crawler) Run(entryURL string, maxEpoch int) { // 信号量:限制同时并发数(使用上方读取的 workers 值) sem := make(chan struct{}, workers) + for _, u := range queue { wg.Add(1) sem <- struct{}{} // 获取一个令牌(阻塞直到有空闲槽位) @@ -347,21 +388,43 @@ func (c *Crawler) Run(entryURL string, maxEpoch int) { // 抓取单个 URL,返回发现的子链接 hrefs := c.visitURL(rawURL) n := len(hrefs) - if n > 0 { - // 每个子链接分得 1/n 的父页面权重 - w := 1.0 / float64(n) - mu.Lock() - for _, h := range hrefs { - if !c.isVisited(h) { - newLinks = append(newLinks, URLWeight{URL: h, Weight: w}) - } - } - mu.Unlock() + if n == 0 { + return } + + // 收集未访问的子链接 + var children []string + for _, h := range hrefs { + if !c.isVisited(h) { + children = append(children, h) + } + } + if len(children) == 0 { + return + } + + // 分配权重 + w := 1.0 / float64(n) + + // 孙链接(来自 priorityChildren)爬取后,子链接进入正常 BFS 队列(不再优先传递) + // 所有子链接统一进入 newLinks,经过 schedule() 调度 + mu.Lock() + for _, h := range children { + newLinks = append(newLinks, URLWeight{URL: h, Weight: w}) + } + mu.Unlock() }(u) } wg.Wait() + // ---- 清空本轮已处理的 priorityQueue ---- + c.priorityChildrenMu.Lock() + if len(priorityQueue) > 0 { + c.priorityChildren = c.priorityChildren[len(priorityQueue):] + // 计数器已在提取时扣除,这里不需要额外操作 + } + c.priorityChildrenMu.Unlock() + // 本轮没有发现新链接,爬取结束 if len(newLinks) == 0 { log.Println("[crawler] empty queue — stopping") diff --git a/dist/assets/index-ClaCiNQl.js b/dist/assets/index-CiQvUT0P.js similarity index 77% rename from dist/assets/index-ClaCiNQl.js rename to dist/assets/index-CiQvUT0P.js index 6eb4bf5..ef63814 100644 --- a/dist/assets/index-ClaCiNQl.js +++ b/dist/assets/index-CiQvUT0P.js @@ -3,4 +3,4 @@ var e=Object.defineProperty,t=(t,n)=>{let r={};for(var i in t)e(r,i,{get:t[i],en `)}getSetCookie(){return this.get(`set-cookie`)||[]}get[Symbol.toStringTag](){return`AxiosHeaders`}static from(e){return e instanceof this?e:new this(e)}static concat(e,...t){let n=new this(e);return t.forEach(e=>n.set(e)),n}static accessor(e){let t=(this[rc]=this[rc]={accessors:{}}).accessors,n=this.prototype;function r(e){let r=ic(e);t[r]||(uc(n,e),t[r]=!0)}return Y.isArray(e)?e.forEach(r):r(e),this}};Q.accessor([`Content-Type`,`Content-Length`,`Accept`,`Accept-Encoding`,`User-Agent`,`Authorization`]),Y.reduceDescriptors(Q.prototype,({value:e},t)=>{let n=t[0].toUpperCase()+t.slice(1);return{get:()=>e,set(e){this[n]=e}}}),Y.freezeMethods(Q);function dc(e,t){let n=this||ec,r=t||n,i=Q.from(r.headers),a=r.data;return Y.forEach(e,function(e){a=e.call(n,a,i.normalize(),t?t.status:void 0)}),i.normalize(),a}function fc(e){return!!(e&&e.__CANCEL__)}var pc=class extends X{constructor(e,t,n){super(e??`canceled`,X.ERR_CANCELED,t,n),this.name=`CanceledError`,this.__CANCEL__=!0}};function mc(e,t,n){let r=n.config.validateStatus;!n.status||!r||r(n.status)?e(n):t(new X(`Request failed with status code `+n.status,[X.ERR_BAD_REQUEST,X.ERR_BAD_RESPONSE][Math.floor(n.status/100)-4],n.config,n.request,n))}function hc(e){let t=/^([-+\w]{1,25})(:?\/\/|:)/.exec(e);return t&&t[1]||``}function gc(e,t){e||=10;let n=Array(e),r=Array(e),i=0,a=0,o;return t=t===void 0?1e3:t,function(s){let c=Date.now(),l=r[a];o||=c,n[i]=s,r[i]=c;let u=a,d=0;for(;u!==i;)d+=n[u++],u%=e;if(i=(i+1)%e,i===a&&(a=(a+1)%e),c-o{n=r,i=null,a&&=(clearTimeout(a),null),e(...t)};return[(...e)=>{let t=Date.now(),s=t-n;s>=r?o(e,t):(i=e,a||=setTimeout(()=>{a=null,o(i)},r-s))},()=>i&&o(i)]}var vc=(e,t,n=3)=>{let r=0,i=gc(50,250);return _c(n=>{let a=n.loaded,o=n.lengthComputable?n.total:void 0,s=a-r,c=i(s),l=a<=o;r=a,e({loaded:a,total:o,progress:o?a/o:void 0,bytes:s,rate:c||void 0,estimated:c&&o&&l?(o-a)/c:void 0,event:n,lengthComputable:o!=null,[t?`download`:`upload`]:!0})},n)},yc=(e,t)=>{let n=e!=null;return[r=>t[0]({lengthComputable:n,total:e,loaded:r}),t[1]]},bc=e=>(...t)=>Y.asap(()=>e(...t)),xc=Z.hasStandardBrowserEnv?((e,t)=>n=>(n=new URL(n,Z.origin),e.protocol===n.protocol&&e.host===n.host&&(t||e.port===n.port)))(new URL(Z.origin),Z.navigator&&/(msie|trident)/i.test(Z.navigator.userAgent)):()=>!0,Sc=Z.hasStandardBrowserEnv?{write(e,t,n,r,i,a,o){if(typeof document>`u`)return;let s=[`${e}=${encodeURIComponent(t)}`];Y.isNumber(n)&&s.push(`expires=${new Date(n).toUTCString()}`),Y.isString(r)&&s.push(`path=${r}`),Y.isString(i)&&s.push(`domain=${i}`),a===!0&&s.push(`secure`),Y.isString(o)&&s.push(`SameSite=${o}`),document.cookie=s.join(`; `)},read(e){if(typeof document>`u`)return null;let t=document.cookie.match(RegExp(`(?:^|; )`+e+`=([^;]*)`));return t?decodeURIComponent(t[1]):null},remove(e){this.write(e,``,Date.now()-864e5,`/`)}}:{write(){},read(){return null},remove(){}};function Cc(e){return typeof e==`string`?/^([a-z][a-z\d+\-.]*:)?\/\//i.test(e):!1}function wc(e,t){return t?e.replace(/\/?\/$/,``)+`/`+t.replace(/^\/+/,``):e}function Tc(e,t,n){let r=!Cc(t);return e&&(r||n==0)?wc(e,t):t}var Ec=e=>e instanceof Q?{...e}:e;function Dc(e,t){t||={};let n={};function r(e,t,n,r){return Y.isPlainObject(e)&&Y.isPlainObject(t)?Y.merge.call({caseless:r},e,t):Y.isPlainObject(t)?Y.merge({},t):Y.isArray(t)?t.slice():t}function i(e,t,n,i){if(!Y.isUndefined(t))return r(e,t,n,i);if(!Y.isUndefined(e))return r(void 0,e,n,i)}function a(e,t){if(!Y.isUndefined(t))return r(void 0,t)}function o(e,t){if(!Y.isUndefined(t))return r(void 0,t);if(!Y.isUndefined(e))return r(void 0,e)}function s(n,i,a){if(a in t)return r(n,i);if(a in e)return r(void 0,n)}let c={url:a,method:a,data:a,baseURL:o,transformRequest:o,transformResponse:o,paramsSerializer:o,timeout:o,timeoutMessage:o,withCredentials:o,withXSRFToken:o,adapter:o,responseType:o,xsrfCookieName:o,xsrfHeaderName:o,onUploadProgress:o,onDownloadProgress:o,decompress:o,maxContentLength:o,maxBodyLength:o,beforeRedirect:o,transport:o,httpAgent:o,httpsAgent:o,cancelToken:o,socketPath:o,responseEncoding:o,validateStatus:s,headers:(e,t,n)=>i(Ec(e),Ec(t),n,!0)};return Y.forEach(Object.keys({...e,...t}),function(r){if(r===`__proto__`||r===`constructor`||r===`prototype`)return;let a=Y.hasOwnProp(c,r)?c[r]:i,o=a(e[r],t[r],r);Y.isUndefined(o)&&a!==s||(n[r]=o)}),n}var Oc=e=>{let t=Dc({},e),{data:n,withXSRFToken:r,xsrfHeaderName:i,xsrfCookieName:a,headers:o,auth:s}=t;if(t.headers=o=Q.from(o),t.url=zs(Tc(t.baseURL,t.url,t.allowAbsoluteUrls),e.params,e.paramsSerializer),s&&o.set(`Authorization`,`Basic `+btoa((s.username||``)+`:`+(s.password?unescape(encodeURIComponent(s.password)):``))),Y.isFormData(n)){if(Z.hasStandardBrowserEnv||Z.hasStandardBrowserWebWorkerEnv)o.setContentType(void 0);else if(Y.isFunction(n.getHeaders)){let e=n.getHeaders(),t=[`content-type`,`content-length`];Object.entries(e).forEach(([e,n])=>{t.includes(e.toLowerCase())&&o.set(e,n)})}}if(Z.hasStandardBrowserEnv&&(r&&Y.isFunction(r)&&(r=r(t)),r||r!==!1&&xc(t.url))){let e=i&&a&&Sc.read(a);e&&o.set(i,e)}return t},kc=typeof XMLHttpRequest<`u`&&function(e){return new Promise(function(t,n){let r=Oc(e),i=r.data,a=Q.from(r.headers).normalize(),{responseType:o,onUploadProgress:s,onDownloadProgress:c}=r,l,u,d,f,p;function m(){f&&f(),p&&p(),r.cancelToken&&r.cancelToken.unsubscribe(l),r.signal&&r.signal.removeEventListener(`abort`,l)}let h=new XMLHttpRequest;h.open(r.method.toUpperCase(),r.url,!0),h.timeout=r.timeout;function g(){if(!h)return;let r=Q.from(`getAllResponseHeaders`in h&&h.getAllResponseHeaders());mc(function(e){t(e),m()},function(e){n(e),m()},{data:!o||o===`text`||o===`json`?h.responseText:h.response,status:h.status,statusText:h.statusText,headers:r,config:e,request:h}),h=null}`onloadend`in h?h.onloadend=g:h.onreadystatechange=function(){!h||h.readyState!==4||h.status===0&&!(h.responseURL&&h.responseURL.indexOf(`file:`)===0)||setTimeout(g)},h.onabort=function(){h&&=(n(new X(`Request aborted`,X.ECONNABORTED,e,h)),null)},h.onerror=function(t){let r=new X(t&&t.message?t.message:`Network Error`,X.ERR_NETWORK,e,h);r.event=t||null,n(r),h=null},h.ontimeout=function(){let t=r.timeout?`timeout of `+r.timeout+`ms exceeded`:`timeout exceeded`,i=r.transitional||Vs;r.timeoutErrorMessage&&(t=r.timeoutErrorMessage),n(new X(t,i.clarifyTimeoutError?X.ETIMEDOUT:X.ECONNABORTED,e,h)),h=null},i===void 0&&a.setContentType(null),`setRequestHeader`in h&&Y.forEach(a.toJSON(),function(e,t){h.setRequestHeader(t,e)}),Y.isUndefined(r.withCredentials)||(h.withCredentials=!!r.withCredentials),o&&o!==`json`&&(h.responseType=r.responseType),c&&([d,p]=vc(c,!0),h.addEventListener(`progress`,d)),s&&h.upload&&([u,f]=vc(s),h.upload.addEventListener(`progress`,u),h.upload.addEventListener(`loadend`,f)),(r.cancelToken||r.signal)&&(l=t=>{h&&=(n(!t||t.type?new pc(null,e,h):t),h.abort(),null)},r.cancelToken&&r.cancelToken.subscribe(l),r.signal&&(r.signal.aborted?l():r.signal.addEventListener(`abort`,l)));let _=hc(r.url);if(_&&Z.protocols.indexOf(_)===-1){n(new X(`Unsupported protocol `+_+`:`,X.ERR_BAD_REQUEST,e));return}h.send(i||null)})},Ac=(e,t)=>{let{length:n}=e=e?e.filter(Boolean):[];if(t||n){let n=new AbortController,r,i=function(e){if(!r){r=!0,o();let t=e instanceof Error?e:this.reason;n.abort(t instanceof X?t:new pc(t instanceof Error?t.message:t))}},a=t&&setTimeout(()=>{a=null,i(new X(`timeout of ${t}ms exceeded`,X.ETIMEDOUT))},t),o=()=>{e&&=(a&&clearTimeout(a),a=null,e.forEach(e=>{e.unsubscribe?e.unsubscribe(i):e.removeEventListener(`abort`,i)}),null)};e.forEach(e=>e.addEventListener(`abort`,i));let{signal:s}=n;return s.unsubscribe=()=>Y.asap(o),s}},jc=function*(e,t){let n=e.byteLength;if(!t||n{let i=Mc(e,t),a=0,o,s=e=>{o||(o=!0,r&&r(e))};return new ReadableStream({async pull(e){try{let{done:t,value:r}=await i.next();if(t){s(),e.close();return}let o=r.byteLength;n&&n(a+=o),e.enqueue(new Uint8Array(r))}catch(e){throw s(e),e}},cancel(e){return s(e),i.return()}},{highWaterMark:2})},Fc=64*1024,{isFunction:Ic}=Y,Lc=(({Request:e,Response:t})=>({Request:e,Response:t}))(Y.global),{ReadableStream:Rc,TextEncoder:zc}=Y.global,Bc=(e,...t)=>{try{return!!e(...t)}catch{return!1}},Vc=e=>{e=Y.merge.call({skipUndefined:!0},Lc,e);let{fetch:t,Request:n,Response:r}=e,i=t?Ic(t):typeof fetch==`function`,a=Ic(n),o=Ic(r);if(!i)return!1;let s=i&&Ic(Rc),c=i&&(typeof zc==`function`?(e=>t=>e.encode(t))(new zc):async e=>new Uint8Array(await new n(e).arrayBuffer())),l=a&&s&&Bc(()=>{let e=!1,t=new Rc,r=new n(Z.origin,{body:t,method:`POST`,get duplex(){return e=!0,`half`}}).headers.has(`Content-Type`);return t.cancel(),e&&!r}),u=o&&s&&Bc(()=>Y.isReadableStream(new r(``).body)),d={stream:u&&(e=>e.body)};i&&[`text`,`arrayBuffer`,`blob`,`formData`,`stream`].forEach(e=>{!d[e]&&(d[e]=(t,n)=>{let r=t&&t[e];if(r)return r.call(t);throw new X(`Response type '${e}' is not supported`,X.ERR_NOT_SUPPORT,n)})});let f=async e=>{if(e==null)return 0;if(Y.isBlob(e))return e.size;if(Y.isSpecCompliantForm(e))return(await new n(Z.origin,{method:`POST`,body:e}).arrayBuffer()).byteLength;if(Y.isArrayBufferView(e)||Y.isArrayBuffer(e))return e.byteLength;if(Y.isURLSearchParams(e)&&(e+=``),Y.isString(e))return(await c(e)).byteLength},p=async(e,t)=>Y.toFiniteNumber(e.getContentLength())??f(t);return async e=>{let{url:i,method:o,data:s,signal:c,cancelToken:f,timeout:m,onDownloadProgress:h,onUploadProgress:g,responseType:_,headers:v,withCredentials:y=`same-origin`,fetchOptions:b}=Oc(e),x=t||fetch;_=_?(_+``).toLowerCase():`text`;let S=Ac([c,f&&f.toAbortSignal()],m),C=null,w=S&&S.unsubscribe&&(()=>{S.unsubscribe()}),ee;try{if(g&&l&&o!==`get`&&o!==`head`&&(ee=await p(v,s))!==0){let e=new n(i,{method:`POST`,body:s,duplex:`half`}),t;if(Y.isFormData(s)&&(t=e.headers.get(`content-type`))&&v.setContentType(t),e.body){let[t,n]=yc(ee,vc(bc(g)));s=Pc(e.body,Fc,t,n)}}Y.isString(y)||(y=y?`include`:`omit`);let t=a&&`credentials`in n.prototype,c={...b,signal:S,method:o.toUpperCase(),headers:v.normalize().toJSON(),body:s,duplex:`half`,credentials:t?y:void 0};C=a&&new n(i,c);let f=await(a?x(C,b):x(i,c)),m=u&&(_===`stream`||_===`response`);if(u&&(h||m&&w)){let e={};[`status`,`statusText`,`headers`].forEach(t=>{e[t]=f[t]});let t=Y.toFiniteNumber(f.headers.get(`content-length`)),[n,i]=h&&yc(t,vc(bc(h),!0))||[];f=new r(Pc(f.body,Fc,n,()=>{i&&i(),w&&w()}),e)}_||=`text`;let te=await d[Y.findKey(d,_)||`text`](f,e);return!m&&w&&w(),await new Promise((t,n)=>{mc(t,n,{data:te,headers:Q.from(f.headers),status:f.status,statusText:f.statusText,config:e,request:C})})}catch(t){throw w&&w(),t&&t.name===`TypeError`&&/Load failed|fetch/i.test(t.message)?Object.assign(new X(`Network Error`,X.ERR_NETWORK,e,C,t&&t.response),{cause:t.cause||t}):X.from(t,t&&t.code,e,C,t&&t.response)}}},Hc=new Map,Uc=e=>{let t=e&&e.env||{},{fetch:n,Request:r,Response:i}=t,a=[r,i,n],o=a.length,s,c,l=Hc;for(;o--;)s=a[o],c=l.get(s),c===void 0&&l.set(s,c=o?new Map:Vc(t)),l=c;return c};Uc();var Wc={http:null,xhr:kc,fetch:{get:Uc}};Y.forEach(Wc,(e,t)=>{if(e){try{Object.defineProperty(e,`name`,{value:t})}catch{}Object.defineProperty(e,`adapterName`,{value:t})}});var Gc=e=>`- ${e}`,Kc=e=>Y.isFunction(e)||e===null||e===!1;function qc(e,t){e=Y.isArray(e)?e:[e];let{length:n}=e,r,i,a={};for(let o=0;o`adapter ${e} `+(t===!1?`is not supported by the environment`:`is not available in the build`));throw new X(`There is no suitable adapter to dispatch the request `+(n?e.length>1?`since : `+e.map(Gc).join(` `):` `+Gc(e[0]):`as no adapter specified`),`ERR_NOT_SUPPORT`)}return i}var Jc={getAdapter:qc,adapters:Wc};function Yc(e){if(e.cancelToken&&e.cancelToken.throwIfRequested(),e.signal&&e.signal.aborted)throw new pc(null,e)}function Xc(e){return Yc(e),e.headers=Q.from(e.headers),e.data=dc.call(e,e.transformRequest),[`post`,`put`,`patch`].indexOf(e.method)!==-1&&e.headers.setContentType(`application/x-www-form-urlencoded`,!1),Jc.getAdapter(e.adapter||ec.adapter,e)(e).then(function(t){return Yc(e),t.data=dc.call(e,e.transformResponse,t),t.headers=Q.from(t.headers),t},function(t){return fc(t)||(Yc(e),t&&t.response&&(t.response.data=dc.call(e,e.transformResponse,t.response),t.response.headers=Q.from(t.response.headers))),Promise.reject(t)})}var Zc=`1.14.0`,Qc={};[`object`,`boolean`,`number`,`function`,`string`,`symbol`].forEach((e,t)=>{Qc[e]=function(n){return typeof n===e||`a`+(t<1?`n `:` `)+e}});var $c={};Qc.transitional=function(e,t,n){function r(e,t){return`[Axios v`+Zc+`] Transitional option '`+e+`'`+t+(n?`. `+n:``)}return(n,i,a)=>{if(e===!1)throw new X(r(i,` has been removed`+(t?` in `+t:``)),X.ERR_DEPRECATED);return t&&!$c[i]&&($c[i]=!0,console.warn(r(i,` has been deprecated since v`+t+` and will be removed in the near future`))),e?e(n,i,a):!0}},Qc.spelling=function(e){return(t,n)=>(console.warn(`${n} is likely a misspelling of ${e}`),!0)};function el(e,t,n){if(typeof e!=`object`)throw new X(`options must be an object`,X.ERR_BAD_OPTION_VALUE);let r=Object.keys(e),i=r.length;for(;i-- >0;){let a=r[i],o=t[a];if(o){let t=e[a],n=t===void 0||o(t,a,e);if(n!==!0)throw new X(`option `+a+` must be `+n,X.ERR_BAD_OPTION_VALUE);continue}if(n!==!0)throw new X(`Unknown option `+a,X.ERR_BAD_OPTION)}}var tl={assertOptions:el,validators:Qc},nl=tl.validators,rl=class{constructor(e){this.defaults=e||{},this.interceptors={request:new Bs,response:new Bs}}async request(e,t){try{return await this._request(e,t)}catch(e){if(e instanceof Error){let t={};Error.captureStackTrace?Error.captureStackTrace(t):t=Error();let n=t.stack?t.stack.replace(/^.+\n/,``):``;try{e.stack?n&&!String(e.stack).endsWith(n.replace(/^.+\n.+\n/,``))&&(e.stack+=` -`+n):e.stack=n}catch{}}throw e}}_request(e,t){typeof e==`string`?(t||={},t.url=e):t=e||{},t=Dc(this.defaults,t);let{transitional:n,paramsSerializer:r,headers:i}=t;n!==void 0&&tl.assertOptions(n,{silentJSONParsing:nl.transitional(nl.boolean),forcedJSONParsing:nl.transitional(nl.boolean),clarifyTimeoutError:nl.transitional(nl.boolean),legacyInterceptorReqResOrdering:nl.transitional(nl.boolean)},!1),r!=null&&(Y.isFunction(r)?t.paramsSerializer={serialize:r}:tl.assertOptions(r,{encode:nl.function,serialize:nl.function},!0)),t.allowAbsoluteUrls!==void 0||(this.defaults.allowAbsoluteUrls===void 0?t.allowAbsoluteUrls=!0:t.allowAbsoluteUrls=this.defaults.allowAbsoluteUrls),tl.assertOptions(t,{baseUrl:nl.spelling(`baseURL`),withXsrfToken:nl.spelling(`withXSRFToken`)},!0),t.method=(t.method||this.defaults.method||`get`).toLowerCase();let a=i&&Y.merge(i.common,i[t.method]);i&&Y.forEach([`delete`,`get`,`head`,`post`,`put`,`patch`,`common`],e=>{delete i[e]}),t.headers=Q.concat(a,i);let o=[],s=!0;this.interceptors.request.forEach(function(e){if(typeof e.runWhen==`function`&&e.runWhen(t)===!1)return;s&&=e.synchronous;let n=t.transitional||Vs;n&&n.legacyInterceptorReqResOrdering?o.unshift(e.fulfilled,e.rejected):o.push(e.fulfilled,e.rejected)});let c=[];this.interceptors.response.forEach(function(e){c.push(e.fulfilled,e.rejected)});let l,u=0,d;if(!s){let e=[Xc.bind(this),void 0];for(e.unshift(...o),e.push(...c),d=e.length,l=Promise.resolve(t);u{if(!n._listeners)return;let t=n._listeners.length;for(;t-- >0;)n._listeners[t](e);n._listeners=null}),this.promise.then=e=>{let t,r=new Promise(e=>{n.subscribe(e),t=e}).then(e);return r.cancel=function(){n.unsubscribe(t)},r},e(function(e,r,i){n.reason||(n.reason=new pc(e,r,i),t(n.reason))})}throwIfRequested(){if(this.reason)throw this.reason}subscribe(e){if(this.reason){e(this.reason);return}this._listeners?this._listeners.push(e):this._listeners=[e]}unsubscribe(e){if(!this._listeners)return;let t=this._listeners.indexOf(e);t!==-1&&this._listeners.splice(t,1)}toAbortSignal(){let e=new AbortController,t=t=>{e.abort(t)};return this.subscribe(t),e.signal.unsubscribe=()=>this.unsubscribe(t),e.signal}static source(){let t;return{token:new e(function(e){t=e}),cancel:t}}};function al(e){return function(t){return e.apply(null,t)}}function ol(e){return Y.isObject(e)&&e.isAxiosError===!0}var sl={Continue:100,SwitchingProtocols:101,Processing:102,EarlyHints:103,Ok:200,Created:201,Accepted:202,NonAuthoritativeInformation:203,NoContent:204,ResetContent:205,PartialContent:206,MultiStatus:207,AlreadyReported:208,ImUsed:226,MultipleChoices:300,MovedPermanently:301,Found:302,SeeOther:303,NotModified:304,UseProxy:305,Unused:306,TemporaryRedirect:307,PermanentRedirect:308,BadRequest:400,Unauthorized:401,PaymentRequired:402,Forbidden:403,NotFound:404,MethodNotAllowed:405,NotAcceptable:406,ProxyAuthenticationRequired:407,RequestTimeout:408,Conflict:409,Gone:410,LengthRequired:411,PreconditionFailed:412,PayloadTooLarge:413,UriTooLong:414,UnsupportedMediaType:415,RangeNotSatisfiable:416,ExpectationFailed:417,ImATeapot:418,MisdirectedRequest:421,UnprocessableEntity:422,Locked:423,FailedDependency:424,TooEarly:425,UpgradeRequired:426,PreconditionRequired:428,TooManyRequests:429,RequestHeaderFieldsTooLarge:431,UnavailableForLegalReasons:451,InternalServerError:500,NotImplemented:501,BadGateway:502,ServiceUnavailable:503,GatewayTimeout:504,HttpVersionNotSupported:505,VariantAlsoNegotiates:506,InsufficientStorage:507,LoopDetected:508,NotExtended:510,NetworkAuthenticationRequired:511,WebServerIsDown:521,ConnectionTimedOut:522,OriginIsUnreachable:523,TimeoutOccurred:524,SslHandshakeFailed:525,InvalidSslCertificate:526};Object.entries(sl).forEach(([e,t])=>{sl[t]=e});function cl(e){let t=new rl(e),n=yo(rl.prototype.request,t);return Y.extend(n,rl.prototype,t,{allOwnKeys:!0}),Y.extend(n,t,null,{allOwnKeys:!0}),n.create=function(t){return cl(Dc(e,t))},n}var $=cl(ec);$.Axios=rl,$.CanceledError=pc,$.CancelToken=il,$.isCancel=fc,$.VERSION=Zc,$.toFormData=Ps,$.AxiosError=X,$.Cancel=$.CanceledError,$.all=function(e){return Promise.all(e)},$.spread=al,$.isAxiosError=ol,$.mergeConfig=Dc,$.AxiosHeaders=Q,$.formToJSON=e=>Qs(Y.isHTMLForm(e)?new FormData(e):e),$.getAdapter=Jc.getAdapter,$.HttpStatusCode=sl,$.default=$;var ll=``;async function ul(e=50){let{data:t}=await $.get(`${ll}/admin/recent`,{params:{limit:e},timeout:15e3});return t}async function dl(){let{data:e}=await $.get(`${ll}/admin/stats`,{timeout:15e3});return e}async function fl(){let{data:e}=await $.get(`${ll}/admin/priority`,{timeout:15e3});return e}async function pl(e){let{data:t}=await $.post(`${ll}/admin/priority`,{url:e},{timeout:15e3});return t}async function ml(e){let{data:t}=await $.delete(`${ll}/admin/priority`,{params:{url:e},timeout:15e3});return t}async function hl(){let{data:e}=await $.post(`${ll}/admin/flush`,null,{timeout:6e4});return e}async function gl(){let{data:e}=await $.get(`${ll}/admin/workers`,{timeout:1e4});return e}async function _l(e){let{data:t}=await $.post(`${ll}/admin/workers`,{workers:e},{timeout:1e4});return t}async function vl(){let{data:e}=await $.get(`${ll}/admin/backlink`,{timeout:1e4});return e}async function yl(){let{data:e}=await $.post(`${ll}/admin/backlink`,null,{timeout:1e4});return e}async function bl(){let{data:e}=await $.get(`${ll}/admin/priority/status`,{timeout:1e4});return e}var xl={class:`p-4 md:p-8`},Sl={key:0,class:`flex items-center justify-center h-48`},Cl={key:1,class:`bg-red-900/30 border border-red-800 rounded-lg p-4 text-red-300`},wl={class:`grid grid-cols-2 md:grid-cols-5 gap-3 md:gap-5 mb-6 md:mb-8`},Tl={class:`bg-gray-900 border border-gray-800 rounded-xl p-5`},El={class:`text-3xl font-bold text-white`},Dl={class:`bg-gray-900 border border-gray-800 rounded-xl p-5`},Ol={class:`text-3xl font-bold text-white`},kl={class:`bg-gray-900 border border-gray-800 rounded-xl p-5`},Al={class:`text-3xl font-bold text-white`},jl={class:`bg-gray-900 border border-gray-800 rounded-xl p-5`},Ml={class:`bg-gray-900 border border-gray-800 rounded-xl p-5 flex flex-col justify-between`},Nl=[`disabled`],Pl={class:`bg-gray-900 border border-gray-800 rounded-xl p-4 md:p-5 mb-6 md:mb-8`},Fl={class:`flex flex-col md:flex-row md:items-center md:justify-between gap-4`},Il={class:`flex items-baseline gap-4`},Ll={class:`text-3xl font-bold text-white`},Rl={key:0,class:`ml-2`},zl={class:`text-3xl font-bold text-orange-400`},Bl={key:0,class:`flex items-center gap-6 text-sm`},Vl={class:`text-center`},Hl={class:`text-center`},Ul={class:`text-center`},Wl={class:`text-2xl font-bold text-gray-400`},Gl={class:`flex items-center gap-2`},Kl=[`disabled`],ql=[`disabled`],Jl=[`disabled`],Yl={key:0,class:`bg-gray-900 border border-gray-800 rounded-xl p-4 md:p-5 mb-6 md:mb-8`},Xl={class:`flex flex-col md:flex-row md:items-center md:justify-between gap-4`},Zl={class:`flex-1`},Ql={class:`grid grid-cols-1 sm:grid-cols-2 gap-3`},$l={class:`text-lg font-bold text-white`},eu=[`title`],tu=[`disabled`],nu={class:`grid grid-cols-1 md:grid-cols-2 gap-4 md:gap-5`},ru={class:`bg-gray-900 border border-gray-800 rounded-xl p-4 md:p-5`},iu={class:`space-y-2`},au=[`title`],ou={class:`flex-1 bg-gray-800 rounded-full h-4 md:h-5 overflow-hidden`},su={class:`w-12 md:w-16 text-xs text-gray-500 text-right shrink-0`},cu={class:`bg-gray-900 border border-gray-800 rounded-xl p-4 md:p-5`},lu={class:`space-y-2`},uu={class:`w-8 md:w-10 text-xs text-gray-400 shrink-0 font-mono`},du={class:`flex-1 bg-gray-800 rounded-full h-4 md:h-5 overflow-hidden`},fu={class:`w-12 md:w-16 text-xs text-gray-500 text-right shrink-0`},pu={__name:`Dashboard`,setup(e){let t=F(null),n=F(!0),r=F(!1),i=F(null),a=null,o=F(0),s=F(0),c=F(0),l=F(!1),u=F(null);ir(async()=>{await Promise.all([d(),f(),m(),h()]),a=setInterval(()=>{d(),f(),m(),h()},5e3)}),cr(()=>{a&&clearInterval(a)});async function d(){try{t.value=await dl(),i.value=null}catch(e){i.value=`无法加载统计数据,可能人服务器未启动或端口不对`,console.error(e)}finally{n.value=!1}}async function f(){try{let e=await gl();o.value=e.configured,s.value=e.active,c.value=e.configured}catch(e){console.error(`Failed to load workers:`,e)}}async function p(){let e=parseInt(c.value,10);if(isNaN(e)||e<1||e>500){i.value=`线程数必须在 1~500 之间`;return}l.value=!0;try{await _l(e),o.value=e,i.value=null}catch(e){i.value=`修改线程数失败: `+e.message}finally{l.value=!1}}async function m(){try{backlinkStatus.value=await vl()}catch(e){console.error(`Failed to load backlink status:`,e)}}async function h(){try{u.value=await bl()}catch(e){console.error(`Failed to load priority status:`,e)}}async function g(){backlinkTriggering.value=!0;try{await yl(),backlinkStatus.value=await vl(),i.value=null}catch(e){i.value=`触发反链计算失败: `+e.message}finally{backlinkTriggering.value=!1}}function _(e){if(!e)return`--`;let t=new Date(e),n=t-new Date,r=Math.floor(n/6e4),i=Math.floor(r/60),a=Math.floor(i/24),o=t.toLocaleTimeString(`zh-CN`,{hour:`2-digit`,minute:`2-digit`}),s=t.toLocaleDateString(`zh-CN`,{month:`short`,day:`numeric`});return a>0?`${s} ${o} (${a}天后)`:i>0?`${s} ${o} (${i}小时后)`:r>0?`${o} (${r}分钟后)`:`${o} (即将执行)`}function v(e){return e?new Date(e).toLocaleString(`zh-CN`,{month:`short`,day:`numeric`,hour:`2-digit`,minute:`2-digit`}):`从未执行`}async function y(){r.value=!0;try{await hl(),t.value=await dl()}catch(e){i.value=`刷盘失败: `+e.message}finally{r.value=!1}}function b(e){return!e&&e!==0?`0`:Number(e).toLocaleString()}function x(e){return e?Object.entries(e).sort((e,t)=>t[1]-e[1]).slice(0,10):[]}function S(e){let t={zh:`#e53e3e`,en:`#3182ce`,ja:`#e53e3e`,ko:`#3182ce`,fr:`#38a169`,de:`#d69e2e`,es:`#38a169`,ru:`#805ad5`,other:`#718096`};return t[e]||t.other}return(e,a)=>(U(),W(`div`,xl,[a[22]||=G(`h1`,{class:`text-xl md:text-2xl font-semibold text-white mb-6 md:mb-8`},`概览`,-1),n.value?(U(),W(`div`,Sl,[...a[3]||=[G(`div`,{class:`text-gray-400 animate-pulse`},`加载中...`,-1)]])):i.value?(U(),W(`div`,Cl,O(i.value),1)):t.value?(U(),W(V,{key:2},[G(`div`,wl,[G(`div`,Tl,[a[4]||=G(`div`,{class:`text-sm text-gray-500 mb-2`},`已爬取 URL`,-1),G(`div`,El,O(b(t.value.total_urls)),1)]),G(`div`,Dl,[a[5]||=G(`div`,{class:`text-sm text-gray-500 mb-2`},`总词数`,-1),G(`div`,Ol,O(b(t.value.total_words)),1)]),G(`div`,kl,[a[6]||=G(`div`,{class:`text-sm text-gray-500 mb-2`},`域名数量`,-1),G(`div`,Al,O(b(t.value.total_domains)),1)]),G(`div`,jl,[a[7]||=G(`div`,{class:`text-sm text-gray-500 mb-2`},`待重爬`,-1),G(`div`,{class:D([`text-3xl font-bold`,t.value.recrawl_eligible>0?`text-orange-400`:`text-green-400`])},O(b(t.value.recrawl_eligible)),3)]),G(`div`,Ml,[G(`div`,null,[a[8]||=G(`div`,{class:`text-sm text-gray-500 mb-2`},`待刷盘`,-1),G(`div`,{class:D([`text-3xl font-bold`,t.value.pending>0?`text-yellow-400`:`text-green-400`])},O(b(t.value.pending)),3)]),G(`button`,{class:`mt-3 w-full bg-blue-700 hover:bg-blue-600 disabled:bg-gray-700 disabled:text-gray-500 text-white text-sm font-medium py-1.5 px-3 rounded transition-colors cursor-pointer`,disabled:r.value||!t.value.pending,onClick:y},O(r.value?`刷盘中...`:`立即刷盘`),9,Nl)])]),G(`div`,Pl,[G(`div`,Fl,[G(`div`,null,[a[13]||=G(`h2`,{class:`text-sm font-semibold text-gray-300 uppercase tracking-wider mb-2`},`爬虫线程`,-1),G(`div`,Il,[G(`div`,null,[a[9]||=G(`div`,{class:`text-xs text-gray-500 mb-0.5`},`实际运行`,-1),G(`div`,{class:D([`text-3xl font-bold`,s.value>0?`text-green-400`:`text-gray-500`])},O(s.value),3)]),a[12]||=G(`div`,{class:`text-gray-700 text-xl`},`/`,-1),G(`div`,null,[a[10]||=G(`div`,{class:`text-xs text-gray-500 mb-0.5`},`设定上限`,-1),G(`div`,Ll,O(o.value),1)]),u.value&&u.value.active>0?(U(),W(`div`,Rl,[a[11]||=G(`div`,{class:`text-xs text-orange-400 mb-0.5`},`优先线程`,-1),G(`div`,zl,`+`+O(u.value.active),1)])):K(``,!0)])]),u.value?(U(),W(`div`,Bl,[G(`div`,Vl,[a[14]||=G(`div`,{class:`text-xs text-gray-500 mb-1`},`待处理队列`,-1),G(`div`,{class:D([`text-2xl font-bold`,u.value.pending>0?`text-yellow-400`:`text-gray-500`])},O(u.value.pending),3)]),G(`div`,Hl,[a[15]||=G(`div`,{class:`text-xs text-gray-500 mb-1`},`正在爬取`,-1),G(`div`,{class:D([`text-2xl font-bold`,u.value.active>0?`text-orange-400`:`text-gray-500`])},O(u.value.active),3)]),G(`div`,Ul,[a[16]||=G(`div`,{class:`text-xs text-gray-500 mb-1`},`上限`,-1),G(`div`,Wl,O(u.value.max_workers),1)])])):K(``,!0),G(`div`,Gl,[G(`button`,{class:`w-9 h-9 flex items-center justify-center rounded-lg bg-gray-800 hover:bg-gray-700 text-gray-300 hover:text-white text-lg font-bold transition-colors cursor-pointer`,onClick:a[0]||=e=>c.value=Math.max(1,c.value-1),disabled:l.value},`−`,8,Kl),An(G(`input`,{type:`number`,"onUpdate:modelValue":a[1]||=e=>c.value=e,min:`1`,max:`500`,class:`w-20 h-9 text-center bg-gray-800 border border-gray-700 rounded-lg text-white text-sm focus:outline-none focus:border-blue-500 transition-colors [appearance:textfield] [&::-webkit-outer-spin-button]:appearance-none [&::-webkit-inner-spin-button]:appearance-none`,onKeyup:fo(p,[`enter`])},null,544),[[oo,c.value,void 0,{number:!0}]]),G(`button`,{class:`w-9 h-9 flex items-center justify-center rounded-lg bg-gray-800 hover:bg-gray-700 text-gray-300 hover:text-white text-lg font-bold transition-colors cursor-pointer`,onClick:a[2]||=e=>c.value=Math.min(500,c.value+1),disabled:l.value},`+`,8,ql),G(`button`,{class:`h-9 px-4 bg-blue-700 hover:bg-blue-600 disabled:bg-gray-700 disabled:text-gray-500 text-white text-sm font-medium rounded-lg transition-colors cursor-pointer`,disabled:l.value||c.value===o.value,onClick:p},O(l.value?`保存中...`:`应用`),9,Jl)])])]),e.backlinkStatus?(U(),W(`div`,Yl,[G(`div`,Xl,[G(`div`,Zl,[a[19]||=G(`h2`,{class:`text-sm font-semibold text-gray-300 uppercase tracking-wider mb-3`},`反链计算(PageRank)`,-1),G(`div`,Ql,[G(`div`,null,[a[17]||=G(`div`,{class:`text-xs text-gray-500 mb-0.5`},`下次自动执行`,-1),G(`div`,{class:D([`text-lg font-bold`,e.backlinkStatus.running?`text-yellow-400`:`text-white`])},O(e.backlinkStatus.running?`计算中...`:_(e.backlinkStatus.next_run)),3)]),G(`div`,null,[a[18]||=G(`div`,{class:`text-xs text-gray-500 mb-0.5`},`上次完成`,-1),G(`div`,$l,O(v(e.backlinkStatus.last_run)),1)])]),e.backlinkStatus.last_error?(U(),W(`div`,{key:0,class:`mt-2 text-xs text-red-400 truncate`,title:e.backlinkStatus.last_error},` 上次错误:`+O(e.backlinkStatus.last_error),9,eu)):K(``,!0)]),G(`button`,{class:`h-9 px-4 bg-blue-700 hover:bg-blue-600 disabled:bg-gray-700 disabled:text-gray-500 text-white text-sm font-medium rounded-lg transition-colors cursor-pointer whitespace-nowrap`,disabled:e.backlinkTriggering||e.backlinkStatus.running,onClick:g},O(e.backlinkTriggering?`已触发...`:e.backlinkStatus.running?`计算中...`:`立即执行`),9,tu)])])):K(``,!0),G(`div`,nu,[G(`div`,ru,[a[20]||=G(`h2`,{class:`text-sm font-semibold text-gray-300 mb-3 md:mb-4 uppercase tracking-wider`},`域名分布 Top 10`,-1),G(`div`,iu,[(U(!0),W(V,null,R(x(t.value.domains),([e,n])=>(U(),W(`div`,{key:e,class:`flex items-center gap-2 md:gap-3`},[G(`div`,{class:`w-24 md:w-36 text-xs text-gray-400 truncate shrink-0`,title:e},O(e),9,au),G(`div`,ou,[G(`div`,{class:`h-full bg-blue-600 rounded-full transition-all duration-500`,style:me({width:`${n/t.value.domains[Object.keys(t.value.domains)[0]]*100}%`})},null,4)]),G(`div`,su,O(b(n)),1)]))),128))])]),G(`div`,cu,[a[21]||=G(`h2`,{class:`text-sm font-semibold text-gray-300 mb-3 md:mb-4 uppercase tracking-wider`},`语种分布`,-1),G(`div`,lu,[(U(!0),W(V,null,R(Object.entries(t.value.languages||{}).sort((e,t)=>t[1]-e[1]),([e,n])=>(U(),W(`div`,{key:e,class:`flex items-center gap-2 md:gap-3`},[G(`div`,uu,O(e),1),G(`div`,du,[G(`div`,{class:`h-full rounded-full transition-all duration-500`,style:me({width:`${n/t.value.total_urls*100}%`,backgroundColor:S(e)})},null,4)]),G(`div`,fu,O(b(n)),1)]))),128))])])])],64)):K(``,!0)]))}},mu={class:`p-4 md:p-8`},hu={class:`flex flex-col md:flex-row md:items-center justify-between mb-4 md:mb-6 gap-3`},gu={class:`text-sm text-gray-500`},_u={class:`flex items-center gap-2 md:gap-3`},vu=[`value`],yu={class:`flex flex-col sm:flex-row items-stretch sm:items-center gap-3 mb-4 md:mb-5`},bu={class:`relative flex-1 max-w-full sm:max-w-sm`},xu={key:0,class:`flex items-center justify-center h-48`},Su={key:1,class:`bg-red-900/30 border border-red-800 rounded-lg p-4 text-red-300`},Cu={key:2,class:`bg-gray-900 border border-gray-800 rounded-xl overflow-hidden`},wu={class:`hidden md:table w-full text-sm`},Tu={class:`px-5 py-3.5`},Eu=[`href`],Du={class:`font-medium text-gray-200 group-hover:text-white line-clamp-2`},Ou={class:`text-xs text-gray-600 mt-0.5 break-all line-clamp-1`},ku={key:0,class:`text-xs text-gray-500 mt-1 line-clamp-1`},Au={class:`px-5 py-3.5`},ju={class:`text-gray-400 text-xs font-mono`},Mu={class:`px-5 py-3.5`},Nu={key:1,class:`text-xs text-gray-600`},Pu={class:`px-5 py-3.5 text-gray-500 text-xs tabular-nums`},Fu={class:`px-5 py-3.5 text-gray-500 text-xs`},Iu={key:0},Lu={class:`md:hidden divide-y divide-gray-800`},Ru=[`href`],zu={class:`font-medium text-gray-200 text-sm line-clamp-2 mb-1`},Bu={class:`text-xs text-gray-500 break-all line-clamp-1 mb-2`},Vu={class:`flex items-center gap-2 text-xs`},Hu={class:`text-gray-400 font-mono`},Uu={class:`text-gray-600 ml-auto`},Wu={key:0,class:`p-8 text-center text-gray-600`},Gu={key:3,class:`mt-3 text-xs text-gray-600 text-right pb-4 md:pb-0`},Ku={__name:`RecentCrawls`,setup(e){let t=F([]),n=F(0),r=F(!0),i=F(null),a=F(``),o=F(``),s=[20,50,100,200],c=F(50);ir(async()=>{await l()});async function l(){r.value=!0,i.value=null;try{let e=await ul(c.value);t.value=e.items||[],n.value=e.total||0}catch(e){i.value=`无法加载数据,可能人服务器未启动`,console.error(e)}finally{r.value=!1}}async function u(e){c.value=e,await l()}let d=ha(()=>{let e=t.value;if(a.value){let t=a.value.toLowerCase();e=e.filter(e=>e.title?.toLowerCase().includes(t)||e.url?.toLowerCase().includes(t)||e.domain?.toLowerCase().includes(t))}return o.value&&(e=e.filter(e=>Object.keys(e.language||{}).includes(o.value))),e});function f(e){return e?new Date(e*1e3).toLocaleString(`zh-CN`,{year:`numeric`,month:`2-digit`,day:`2-digit`,hour:`2-digit`,minute:`2-digit`,second:`2-digit`,hour12:!1}):`-`}function p(e){return{zh:{label:`中文`,cls:`bg-red-900/60 text-red-300`},en:{label:`EN`,cls:`bg-blue-900/60 text-blue-300`},ja:{label:`日`,cls:`bg-pink-900/60 text-pink-300`},ko:{label:`한`,cls:`bg-blue-900/60 text-blue-300`},fr:{label:`FR`,cls:`bg-green-900/60 text-green-300`},de:{label:`DE`,cls:`bg-yellow-900/60 text-yellow-300`},es:{label:`ES`,cls:`bg-green-900/60 text-green-300`},ru:{label:`RU`,cls:`bg-purple-900/60 text-purple-300`}}[e]||{label:e,cls:`bg-gray-800 text-gray-400`}}function m(e){return e?Object.entries(e).sort((e,t)=>t[1]-e[1])[0]:null}return(e,t)=>(U(),W(`div`,mu,[G(`div`,hu,[G(`div`,null,[t[4]||=G(`h1`,{class:`text-xl md:text-2xl font-semibold text-white mb-1`},`最近爬取`,-1),G(`p`,gu,`共 `+O(n.value.toLocaleString())+` 条记录`,1)]),G(`div`,_u,[An(G(`select`,{"onUpdate:modelValue":t[0]||=e=>c.value=e,onChange:t[1]||=e=>u(c.value),class:`bg-gray-900 border border-gray-700 text-gray-300 text-sm rounded-lg px-3 py-2 focus:border-blue-500 focus:outline-none`},[(U(),W(V,null,R(s,e=>G(`option`,{key:e,value:e},`显示 `+O(e)+` 条`,9,vu)),64))],544),[[so,c.value]]),G(`button`,{onClick:l,class:`bg-blue-600 hover:bg-blue-700 text-white text-sm px-4 py-2 rounded-lg transition-colors`},` 刷新 `)])]),G(`div`,yu,[G(`div`,bu,[An(G(`input`,{"onUpdate:modelValue":t[2]||=e=>a.value=e,type:`text`,placeholder:`搜索标题、URL、域名...`,class:`w-full bg-gray-900 border border-gray-700 text-gray-200 text-sm rounded-lg pl-10 pr-4 py-2 focus:border-blue-500 focus:outline-none placeholder-gray-600`},null,512),[[oo,a.value]]),t[5]||=G(`span`,{class:`absolute left-3 top-1/2 -translate-y-1/2 text-gray-500`},`🔍`,-1)]),An(G(`select`,{"onUpdate:modelValue":t[3]||=e=>o.value=e,class:`bg-gray-900 border border-gray-700 text-gray-300 text-sm rounded-lg px-3 py-2 focus:border-blue-500 focus:outline-none`},[...t[6]||=[Ui(``,9)]],512),[[so,o.value]])]),r.value?(U(),W(`div`,xu,[...t[7]||=[G(`div`,{class:`text-gray-400 animate-pulse`},`加载中...`,-1)]])):i.value?(U(),W(`div`,Su,O(i.value),1)):(U(),W(`div`,Cu,[G(`table`,wu,[t[9]||=G(`thead`,null,[G(`tr`,{class:`border-b border-gray-800`},[G(`th`,{class:`text-left px-5 py-3 text-gray-500 font-medium text-xs uppercase tracking-wider`},`标题`),G(`th`,{class:`text-left px-5 py-3 text-gray-500 font-medium text-xs uppercase tracking-wider w-28`},`域名`),G(`th`,{class:`text-left px-5 py-3 text-gray-500 font-medium text-xs uppercase tracking-wider w-16`},`语种`),G(`th`,{class:`text-left px-5 py-3 text-gray-500 font-medium text-xs uppercase tracking-wider w-20`},`字数`),G(`th`,{class:`text-left px-5 py-3 text-gray-500 font-medium text-xs uppercase tracking-wider w-48`},`爬取时间`)])],-1),G(`tbody`,null,[(U(!0),W(V,null,R(d.value,e=>(U(),W(`tr`,{key:e.url,class:`border-b border-gray-800/50 hover:bg-gray-800/40 transition-colors group`},[G(`td`,Tu,[G(`a`,{href:e.url,target:`_blank`,rel:`noopener noreferrer`,class:`block hover:opacity-80 transition-opacity`},[G(`div`,Du,O(e.title||`(无标题)`),1),G(`div`,Ou,O(e.url),1)],8,Eu),e.description?(U(),W(`div`,ku,O(e.description),1)):K(``,!0)]),G(`td`,Au,[G(`span`,ju,O(e.domain),1)]),G(`td`,Mu,[m(e.language)?(U(),W(`span`,{key:0,class:D([`text-xs px-1.5 py-0.5 rounded font-medium`,p(m(e.language)[0]).cls])},O(p(m(e.language)[0]).label)+` `+O((m(e.language)[1]*100).toFixed(0))+`% `,3)):(U(),W(`span`,Nu,`-`))]),G(`td`,Pu,O(e.word_count.toLocaleString()),1),G(`td`,Fu,O(f(e.crawled_at)),1)]))),128)),d.value.length?K(``,!0):(U(),W(`tr`,Iu,[...t[8]||=[G(`td`,{colspan:`5`,class:`px-5 py-12 text-center text-gray-600`},` 没有找到匹配的记录 `,-1)]]))])]),G(`div`,Lu,[(U(!0),W(V,null,R(d.value,e=>(U(),W(`a`,{key:e.url,href:e.url,target:`_blank`,rel:`noopener noreferrer`,class:`block p-4 hover:bg-gray-800/40 transition-colors`},[G(`div`,zu,O(e.title||`(无标题)`),1),G(`div`,Bu,O(e.url),1),G(`div`,Vu,[G(`span`,Hu,O(e.domain),1),m(e.language)?(U(),W(`span`,{key:0,class:D([`px-1.5 py-0.5 rounded font-medium`,p(m(e.language)[0]).cls])},O(p(m(e.language)[0]).label),3)):K(``,!0),G(`span`,Uu,O(f(e.crawled_at)),1)])],8,Ru))),128)),d.value.length?K(``,!0):(U(),W(`div`,Wu,` 没有找到匹配的记录 `))])])),!r.value&&!i.value&&d.value.length?(U(),W(`div`,Gu,` 筛选后 `+O(d.value.length)+` 条 / 共 `+O(n.value.toLocaleString())+` 条 `,1)):K(``,!0)]))}},qu={class:`p-4 md:p-8`},Ju={class:`bg-gray-900 rounded-xl p-4 md:p-6 mb-4 md:mb-6 border border-gray-800`},Yu={class:`flex flex-col sm:flex-row gap-3`},Xu=[`disabled`],Zu=[`disabled`],Qu={key:0,class:`mt-3 text-sm text-red-400`},$u={key:1,class:`mt-3 text-sm text-green-400`},ed={class:`bg-gray-900 rounded-xl border border-gray-800`},td={class:`px-4 md:px-6 py-3 md:py-4 border-b border-gray-800 flex items-center justify-between`},nd={class:`text-xs text-gray-500`},rd={key:0,class:`p-8 text-center text-gray-500 text-sm`},id={key:1,class:`p-8 text-center text-red-400 text-sm`},ad={key:2,class:`p-8 text-center text-gray-500 text-sm`},od={key:3,class:`hidden md:table w-full text-sm`},sd={class:`divide-y divide-gray-800`},cd={class:`px-6 py-3`},ld={class:`text-gray-300 break-all`},ud={class:`px-6 py-3`},dd={key:0,class:`inline-block px-2 py-0.5 text-xs rounded bg-purple-900 text-purple-300`},fd={key:1,class:`inline-block px-2 py-0.5 text-xs rounded bg-blue-900 text-blue-300`},pd={class:`px-6 py-3 text-gray-500`},md={class:`px-6 py-3`},hd=[`onClick`],gd={class:`md:hidden divide-y divide-gray-800`},_d={class:`flex-1 min-w-0`},vd={class:`text-gray-300 text-sm break-all mb-1`},yd={class:`flex items-center gap-2`},bd={key:0,class:`inline-block px-2 py-0.5 text-xs rounded bg-purple-900 text-purple-300`},xd={key:1,class:`inline-block px-2 py-0.5 text-xs rounded bg-blue-900 text-blue-300`},Sd={class:`text-xs text-gray-500`},Cd=[`onClick`],wd={__name:`PriorityCrawl`,setup(e){let t=F([]),n=F(!0),r=F(null),i=F(!1),a=F(null),o=F(!1),s=F(``);async function c(){n.value=!0,r.value=null;try{t.value=(await fl()).items||[]}catch(e){r.value=`加载失败,请检查人服务器是否启动`,console.error(e)}finally{n.value=!1}}async function l(){let e=s.value.trim();if(e){i.value=!0,a.value=null,o.value=!1;try{await pl(e),s.value=``,o.value=!0,setTimeout(()=>{o.value=!1},3e3),await c()}catch(e){a.value=e?.response?.data?.error||`添加失败`}finally{i.value=!1}}}async function u(e){try{await ml(e),await c()}catch(e){r.value=`删除失败`,console.error(e)}}function d(e){return e?new Date(e*1e3).toLocaleString(`zh-CN`,{month:`2-digit`,day:`2-digit`,hour:`2-digit`,minute:`2-digit`}):`-`}return ir(c),(e,c)=>(U(),W(`div`,qu,[c[3]||=G(`h1`,{class:`text-xl md:text-2xl font-semibold text-white mb-2`},`插入爬取`,-1),c[4]||=G(`p`,{class:`text-sm text-gray-500 mb-6 md:mb-8`},` 添加 URL 或域名,下一轮爬取时会优先抓取。纯域名会自动补全为 https://www.域名/。 `,-1),G(`div`,Ju,[G(`div`,Yu,[An(G(`input`,{"onUpdate:modelValue":c[0]||=e=>s.value=e,onKeyup:fo(l,[`enter`]),type:`text`,placeholder:`输入 URL 或域名,例如 https://example.com 或 example.com`,class:`flex-1 bg-gray-800 border border-gray-700 rounded-lg px-4 py-2.5 text-gray-100 placeholder-gray-500 focus:outline-none focus:border-blue-500 focus:ring-1 focus:ring-blue-500 text-sm`,disabled:i.value},null,40,Xu),[[oo,s.value]]),G(`button`,{onClick:l,disabled:i.value||!s.value.trim(),class:`px-6 py-2.5 bg-blue-600 hover:bg-blue-500 disabled:bg-gray-700 disabled:text-gray-500 text-white text-sm font-medium rounded-lg transition-colors cursor-pointer whitespace-nowrap`},O(i.value?`添加中...`:`插入队列`),9,Zu)]),a.value?(U(),W(`div`,Qu,O(a.value),1)):K(``,!0),o.value?(U(),W(`div`,$u,` 已添加到优先队列,将在下一轮爬取时优先抓取 `)):K(``,!0)]),G(`div`,ed,[G(`div`,td,[c[1]||=G(`span`,{class:`text-sm font-medium text-gray-300`},`待爬取队列`,-1),G(`span`,nd,O(t.value.length)+` 条`,1)]),n.value?(U(),W(`div`,rd,` 加载中... `)):r.value?(U(),W(`div`,id,O(r.value),1)):t.value.length===0?(U(),W(`div`,ad,` 暂无待爬取的优先 URL `)):(U(),W(`table`,od,[c[2]||=G(`thead`,null,[G(`tr`,{class:`text-left text-gray-500 text-xs border-b border-gray-800`},[G(`th`,{class:`px-6 py-3 font-medium`},`URL`),G(`th`,{class:`px-6 py-3 font-medium w-28`},`类型`),G(`th`,{class:`px-6 py-3 font-medium w-40`},`添加时间`),G(`th`,{class:`px-6 py-3 font-medium w-16`},`操作`)])],-1),G(`tbody`,sd,[(U(!0),W(V,null,R(t.value,e=>(U(),W(`tr`,{key:e.url,class:`hover:bg-gray-800/50`},[G(`td`,cd,[G(`span`,ld,O(e.url),1)]),G(`td`,ud,[e.domain?(U(),W(`span`,dd,`域名`)):(U(),W(`span`,fd,`URL`))]),G(`td`,pd,O(d(e.added_at)),1),G(`td`,md,[G(`button`,{onClick:t=>u(e.url),class:`text-red-400 hover:text-red-300 text-xs cursor-pointer`},` 删除 `,8,hd)])]))),128))])])),G(`div`,gd,[(U(!0),W(V,null,R(t.value,e=>(U(),W(`div`,{key:e.url,class:`p-4 hover:bg-gray-800/50 flex items-center justify-between gap-3`},[G(`div`,_d,[G(`div`,vd,O(e.url),1),G(`div`,yd,[e.domain?(U(),W(`span`,bd,`域名`)):(U(),W(`span`,xd,`URL`)),G(`span`,Sd,O(d(e.added_at)),1)])]),G(`button`,{onClick:t=>u(e.url),class:`text-red-400 hover:text-red-300 text-xs cursor-pointer shrink-0 px-2 py-1`},` 删除 `,8,Cd)]))),128))])])]))}},Td={class:`flex flex-col h-full`},Ed={class:`bg-gray-950 border-b border-gray-800 px-4 md:px-8 py-4 md:py-6`},Dd={class:`max-w-3xl mx-auto`},Od={class:`relative`},kd={class:`flex-1 overflow-y-auto px-8 py-6`},Ad={class:`max-w-3xl mx-auto`},jd={key:0,class:`flex items-center gap-3 text-gray-400 py-8`},Md={key:1,class:`bg-red-900/30 border border-red-800 rounded-lg p-4 text-red-300`},Nd={key:2,class:`py-16 text-center text-gray-600`},Pd={key:3,class:`py-16 text-center text-gray-600`},Fd={key:4,class:`flex flex-col sm:flex-row sm:items-center gap-2 sm:gap-4 mb-4 md:mb-5 text-sm text-gray-500`},Id={class:`flex items-center gap-2`},Ld={class:`text-gray-300`},Rd={class:`flex flex-wrap gap-2 sm:ml-auto`},zd={class:`text-gray-300`},Bd={key:5,class:`space-y-2 md:space-y-1`},Vd=[`onClick`],Hd={class:`flex items-start gap-2 md:gap-3 mb-2`},Ud={class:`flex-1 min-w-0`},Wd={class:`text-blue-400 group-hover:text-blue-300 text-base md:text-lg leading-snug`},Gd={class:`text-xs text-gray-600 mt-0.5 truncate`},Kd={class:`flex flex-col items-end gap-1 shrink-0`},qd={class:`text-xs text-gray-600`},Jd={class:`w-12 md:w-14 bg-gray-800 rounded-full h-1.5 overflow-hidden`},Yd={class:`text-gray-400 text-sm leading-relaxed mb-2 md:mb-3`},Xd={class:`flex flex-wrap items-center gap-2 md:gap-3 text-xs`},Zd={class:`flex flex-wrap gap-1.5`},Qd=[`title`],$d={class:`text-blue-400`},ef={class:`text-gray-600 ml-1`},tf={key:0,class:`text-gray-600`},nf={class:`text-gray-700 ml-0 md:ml-auto`},rf={class:`text-xs text-gray-700 mt-1 truncate`},af={key:6,class:`mt-6 text-center`},of=``,sf=10,cf={__name:`SearchView`,setup(e){let t=F(``),n=F([]),r=F(0),i=F({}),a=F(!1),o=F(null),s=F(0),c=null;async function l(e,t=!1){if(!e.trim()){n.value=[],r.value=0,i.value={};return}t||(s.value=0),a.value=!0,o.value=null;try{let a=encodeURIComponent(e),o=s.value*sf,c=await fetch(`${of}/search?qh=${a}&slice=${o}:${o+sf}`);if(!c.ok)throw Error(`HTTP ${c.status}`);let l=await c.json();t?n.value=[...n.value,...l.results||[]]:n.value=l.results||[],r.value=l.total||0,i.value=l.counts||{}}catch(e){o.value=`搜索失败,可能是人服务器未启动`,console.error(e)}finally{a.value=!1}}function u(){clearTimeout(c),c=setTimeout(()=>l(t.value),400)}function d(e){e.key===`Enter`&&(clearTimeout(c),l(t.value))}function f(e,t=120){return e?e.length>t?e.slice(0,t)+`…`:e:``}function p(e){return e?e.toLocaleString():`0`}function m(e){return Math.min(100,Math.round(e*100))}function h(e){return{zh:`中文`,en:`EN`,ja:`JA`,ko:`KO`,fr:`FR`,de:`DE`,es:`ES`,ru:`RU`}[e]||e?.toUpperCase()}function g(e){try{return new URL(e).hostname}catch{return e}}function _(e){window.open(e,`_blank`)}return(e,c)=>(U(),W(`div`,Td,[G(`div`,Ed,[G(`div`,Dd,[G(`div`,Od,[An(G(`input`,{"onUpdate:modelValue":c[0]||=e=>t.value=e,onInput:u,onKeydown:d,type:`text`,placeholder:`输入关键词搜索,或用 site:example.com 限定域名`,class:`w-full bg-gray-900 border border-gray-700 rounded-xl md:rounded-2xl px-4 md:px-6 py-3 md:py-4 pr-20 md:pr-14 text-white text-base md:text-lg placeholder-gray-600 focus:outline-none focus:border-blue-500 focus:ring-1 focus:ring-blue-500 transition`},null,544),[[oo,t.value]]),G(`button`,{onClick:c[1]||=e=>l(t.value),class:`absolute right-2 md:right-3 top-1/2 -translate-y-1/2 bg-blue-600 hover:bg-blue-500 text-white rounded-lg md:rounded-xl px-3 md:px-4 py-1.5 md:py-2 text-sm font-medium transition`},` 搜索 `)])])]),G(`div`,kd,[G(`div`,Ad,[a.value?(U(),W(`div`,jd,[...c[3]||=[G(`div`,{class:`w-5 h-5 border-2 border-blue-400 border-t-transparent rounded-full animate-spin`},null,-1),G(`span`,null,`搜索中...`,-1)]])):o.value?(U(),W(`div`,Md,O(o.value),1)):t.value.trim()?n.value.length===0&&!a.value?(U(),W(`div`,Pd,` 未找到相关结果 `)):n.value.length>0?(U(),W(`div`,Fd,[G(`div`,Id,[G(`span`,null,[c[4]||=Hi(`找到约 `,-1),G(`strong`,Ld,O(p(r.value)),1),c[5]||=Hi(` 条结果`,-1)]),c[6]||=G(`span`,{class:`text-gray-700`},`|`,-1),G(`span`,null,O(n.value.length)+` 条已加载`,1)]),G(`div`,Rd,[(U(!0),W(V,null,R(i.value,(e,t)=>(U(),W(`span`,{key:t,class:`inline-flex items-center gap-1 bg-gray-800 rounded px-2 py-0.5 text-xs text-gray-400`},[G(`span`,zd,O(t),1),G(`span`,null,O(p(e)),1)]))),128))])])):K(``,!0):(U(),W(`div`,Nd,` 输入关键词开始搜索 `)),n.value.length>0?(U(),W(`div`,Bd,[(U(!0),W(V,null,R(n.value,(e,t)=>(U(),W(`div`,{key:t,onClick:t=>_(e.url),class:`group block bg-gray-900/50 hover:bg-gray-900 border border-gray-800 hover:border-gray-700 rounded-xl p-4 md:p-5 cursor-pointer transition`},[G(`div`,Hd,[G(`div`,Ud,[G(`div`,Wd,O(e.snippet?.title||g(e.url)),1),G(`div`,Gd,O(g(e.url)),1)]),G(`div`,Kd,[G(`div`,qd,O(m(e.score))+`%`,1),G(`div`,Jd,[G(`div`,{class:`h-full bg-blue-500 rounded-full`,style:me({width:m(e.score)+`%`})},null,4)])])]),G(`p`,Yd,O(f(e.snippet?.description||e.snippet?.text)),1),G(`div`,Xd,[G(`div`,Zd,[(U(!0),W(V,null,R(e.relevance,(e,t)=>(U(),W(`span`,{key:t,class:`inline-flex items-center bg-gray-800 rounded px-1.5 py-0.5`,title:`${t}: ${(e*100).toFixed(1)}%`},[G(`span`,$d,O(t),1),G(`span`,ef,O((e*100).toFixed(0))+`%`,1)],8,Qd))),128))]),e.snippet?.language?(U(),W(`span`,tf,O(h(e.snippet.language)),1)):K(``,!0),G(`span`,nf,O(p(e.domain_count))+` 个结果`,1)]),G(`div`,rf,O(e.url),1)],8,Vd))),128))])):K(``,!0),n.value.length>0&&n.value.length{s.value++,l(t.value,!0)},class:`bg-gray-800 hover:bg-gray-700 text-gray-300 rounded-xl px-6 py-2.5 text-sm transition`},` 加载更多 (`+O(n.value.length)+`/`+O(p(r.value))+`) `,1)])):K(``,!0)])])]))}},lf={class:`flex h-screen bg-gray-950 text-gray-100 font-sans`},uf={class:`hidden md:flex w-56 bg-gray-900 border-r border-gray-800 flex-col shrink-0`},df={class:`flex-1 py-4 px-3 space-y-1`},ff=[`onClick`],pf={class:`flex-1 overflow-y-auto pb-16 md:pb-0`},mf={class:`md:hidden fixed bottom-0 left-0 right-0 bg-gray-900 border-t border-gray-800 flex justify-around items-center h-16 z-50`},hf=[`onClick`],gf={class:`text-lg`},_f={class:`text-[10px]`};go({__name:`App`,setup(e){let t=F(`dashboard`),n=[{id:`dashboard`,label:`概览`,icon:`📊`},{id:`recent`,label:`最近`,icon:`🕷️`},{id:`search`,label:`搜索`,icon:`🔍`},{id:`priority`,label:`插入`,icon:`🚀`}];return(e,r)=>(U(),W(`div`,lf,[G(`aside`,uf,[r[0]||=G(`div`,{class:`px-5 py-5 border-b border-gray-800`},[G(`div`,{class:`text-lg font-semibold text-white tracking-tight`},`SESE Admin`),G(`div`,{class:`text-xs text-gray-500 mt-0.5`},`爬取内容监控`)],-1),G(`nav`,df,[(U(),W(V,null,R(n,e=>G(`button`,{key:e.id,onClick:n=>t.value=e.id,class:D([`w-full flex items-center gap-3 px-3 py-2.5 rounded-lg text-sm font-medium transition-colors`,t.value===e.id?`bg-blue-600 text-white`:`text-gray-400 hover:text-white hover:bg-gray-800`])},[G(`span`,null,O(e.icon),1),Hi(` `+O(e.label),1)],10,ff)),64))]),r[1]||=G(`div`,{class:`px-5 py-4 border-t border-gray-800`},[G(`div`,{class:`text-xs text-gray-600`},`sese-engine v1.0`)],-1)]),G(`main`,pf,[t.value===`dashboard`?(U(),Ni(pu,{key:0})):t.value===`recent`?(U(),Ni(Ku,{key:1})):t.value===`search`?(U(),Ni(cf,{key:2})):t.value===`priority`?(U(),Ni(wd,{key:3})):K(``,!0)]),G(`nav`,mf,[(U(),W(V,null,R(n,e=>G(`button`,{key:e.id,onClick:n=>t.value=e.id,class:D([`flex flex-col items-center justify-center gap-0.5 flex-1 h-full transition-colors`,t.value===e.id?`text-blue-400`:`text-gray-500`])},[G(`span`,gf,O(e.icon),1),G(`span`,_f,O(e.label),1)],10,hf)),64))])]))}}).mount(`#app`); \ No newline at end of file +`+n):e.stack=n}catch{}}throw e}}_request(e,t){typeof e==`string`?(t||={},t.url=e):t=e||{},t=Dc(this.defaults,t);let{transitional:n,paramsSerializer:r,headers:i}=t;n!==void 0&&tl.assertOptions(n,{silentJSONParsing:nl.transitional(nl.boolean),forcedJSONParsing:nl.transitional(nl.boolean),clarifyTimeoutError:nl.transitional(nl.boolean),legacyInterceptorReqResOrdering:nl.transitional(nl.boolean)},!1),r!=null&&(Y.isFunction(r)?t.paramsSerializer={serialize:r}:tl.assertOptions(r,{encode:nl.function,serialize:nl.function},!0)),t.allowAbsoluteUrls!==void 0||(this.defaults.allowAbsoluteUrls===void 0?t.allowAbsoluteUrls=!0:t.allowAbsoluteUrls=this.defaults.allowAbsoluteUrls),tl.assertOptions(t,{baseUrl:nl.spelling(`baseURL`),withXsrfToken:nl.spelling(`withXSRFToken`)},!0),t.method=(t.method||this.defaults.method||`get`).toLowerCase();let a=i&&Y.merge(i.common,i[t.method]);i&&Y.forEach([`delete`,`get`,`head`,`post`,`put`,`patch`,`common`],e=>{delete i[e]}),t.headers=Q.concat(a,i);let o=[],s=!0;this.interceptors.request.forEach(function(e){if(typeof e.runWhen==`function`&&e.runWhen(t)===!1)return;s&&=e.synchronous;let n=t.transitional||Vs;n&&n.legacyInterceptorReqResOrdering?o.unshift(e.fulfilled,e.rejected):o.push(e.fulfilled,e.rejected)});let c=[];this.interceptors.response.forEach(function(e){c.push(e.fulfilled,e.rejected)});let l,u=0,d;if(!s){let e=[Xc.bind(this),void 0];for(e.unshift(...o),e.push(...c),d=e.length,l=Promise.resolve(t);u{if(!n._listeners)return;let t=n._listeners.length;for(;t-- >0;)n._listeners[t](e);n._listeners=null}),this.promise.then=e=>{let t,r=new Promise(e=>{n.subscribe(e),t=e}).then(e);return r.cancel=function(){n.unsubscribe(t)},r},e(function(e,r,i){n.reason||(n.reason=new pc(e,r,i),t(n.reason))})}throwIfRequested(){if(this.reason)throw this.reason}subscribe(e){if(this.reason){e(this.reason);return}this._listeners?this._listeners.push(e):this._listeners=[e]}unsubscribe(e){if(!this._listeners)return;let t=this._listeners.indexOf(e);t!==-1&&this._listeners.splice(t,1)}toAbortSignal(){let e=new AbortController,t=t=>{e.abort(t)};return this.subscribe(t),e.signal.unsubscribe=()=>this.unsubscribe(t),e.signal}static source(){let t;return{token:new e(function(e){t=e}),cancel:t}}};function al(e){return function(t){return e.apply(null,t)}}function ol(e){return Y.isObject(e)&&e.isAxiosError===!0}var sl={Continue:100,SwitchingProtocols:101,Processing:102,EarlyHints:103,Ok:200,Created:201,Accepted:202,NonAuthoritativeInformation:203,NoContent:204,ResetContent:205,PartialContent:206,MultiStatus:207,AlreadyReported:208,ImUsed:226,MultipleChoices:300,MovedPermanently:301,Found:302,SeeOther:303,NotModified:304,UseProxy:305,Unused:306,TemporaryRedirect:307,PermanentRedirect:308,BadRequest:400,Unauthorized:401,PaymentRequired:402,Forbidden:403,NotFound:404,MethodNotAllowed:405,NotAcceptable:406,ProxyAuthenticationRequired:407,RequestTimeout:408,Conflict:409,Gone:410,LengthRequired:411,PreconditionFailed:412,PayloadTooLarge:413,UriTooLong:414,UnsupportedMediaType:415,RangeNotSatisfiable:416,ExpectationFailed:417,ImATeapot:418,MisdirectedRequest:421,UnprocessableEntity:422,Locked:423,FailedDependency:424,TooEarly:425,UpgradeRequired:426,PreconditionRequired:428,TooManyRequests:429,RequestHeaderFieldsTooLarge:431,UnavailableForLegalReasons:451,InternalServerError:500,NotImplemented:501,BadGateway:502,ServiceUnavailable:503,GatewayTimeout:504,HttpVersionNotSupported:505,VariantAlsoNegotiates:506,InsufficientStorage:507,LoopDetected:508,NotExtended:510,NetworkAuthenticationRequired:511,WebServerIsDown:521,ConnectionTimedOut:522,OriginIsUnreachable:523,TimeoutOccurred:524,SslHandshakeFailed:525,InvalidSslCertificate:526};Object.entries(sl).forEach(([e,t])=>{sl[t]=e});function cl(e){let t=new rl(e),n=yo(rl.prototype.request,t);return Y.extend(n,rl.prototype,t,{allOwnKeys:!0}),Y.extend(n,t,null,{allOwnKeys:!0}),n.create=function(t){return cl(Dc(e,t))},n}var $=cl(ec);$.Axios=rl,$.CanceledError=pc,$.CancelToken=il,$.isCancel=fc,$.VERSION=Zc,$.toFormData=Ps,$.AxiosError=X,$.Cancel=$.CanceledError,$.all=function(e){return Promise.all(e)},$.spread=al,$.isAxiosError=ol,$.mergeConfig=Dc,$.AxiosHeaders=Q,$.formToJSON=e=>Qs(Y.isHTMLForm(e)?new FormData(e):e),$.getAdapter=Jc.getAdapter,$.HttpStatusCode=sl,$.default=$;var ll=``;async function ul(e=50){let{data:t}=await $.get(`${ll}/admin/recent`,{params:{limit:e},timeout:15e3});return t}async function dl(){let{data:e}=await $.get(`${ll}/admin/stats`,{timeout:15e3});return e}async function fl(){let{data:e}=await $.get(`${ll}/admin/priority`,{timeout:15e3});return e}async function pl(e){let{data:t}=await $.post(`${ll}/admin/priority`,{url:e},{timeout:15e3});return t}async function ml(e){let{data:t}=await $.delete(`${ll}/admin/priority`,{params:{url:e},timeout:15e3});return t}async function hl(){let{data:e}=await $.post(`${ll}/admin/flush`,null,{timeout:6e4});return e}async function gl(){let{data:e}=await $.get(`${ll}/admin/workers`,{timeout:1e4});return e}async function _l(e){let{data:t}=await $.post(`${ll}/admin/workers`,{workers:e},{timeout:1e4});return t}async function vl(){let{data:e}=await $.get(`${ll}/admin/backlink`,{timeout:1e4});return e}async function yl(){let{data:e}=await $.post(`${ll}/admin/backlink`,null,{timeout:1e4});return e}async function bl(){let{data:e}=await $.get(`${ll}/admin/priority/status`,{timeout:1e4});return e}var xl={class:`p-4 md:p-8`},Sl={key:0,class:`flex items-center justify-center h-48`},Cl={key:1,class:`bg-red-900/30 border border-red-800 rounded-lg p-4 text-red-300`},wl={class:`grid grid-cols-2 md:grid-cols-5 gap-3 md:gap-5 mb-6 md:mb-8`},Tl={class:`bg-gray-900 border border-gray-800 rounded-xl p-5`},El={class:`text-3xl font-bold text-white`},Dl={class:`bg-gray-900 border border-gray-800 rounded-xl p-5`},Ol={class:`text-3xl font-bold text-white`},kl={class:`bg-gray-900 border border-gray-800 rounded-xl p-5`},Al={class:`text-3xl font-bold text-white`},jl={class:`bg-gray-900 border border-gray-800 rounded-xl p-5`},Ml={class:`bg-gray-900 border border-gray-800 rounded-xl p-5 flex flex-col justify-between`},Nl=[`disabled`],Pl={class:`bg-gray-900 border border-gray-800 rounded-xl p-4 md:p-5 mb-6 md:mb-8`},Fl={class:`flex flex-col md:flex-row md:items-center md:justify-between gap-4`},Il={class:`flex items-baseline gap-4`},Ll={class:`text-3xl font-bold text-white`},Rl={key:0,class:`ml-2`},zl={class:`text-3xl font-bold text-orange-400`},Bl={key:0,class:`flex items-center gap-6 text-sm`},Vl={class:`text-center`},Hl={class:`text-center`},Ul={class:`text-center`},Wl={class:`text-center`},Gl={class:`text-2xl font-bold text-gray-400`},Kl={class:`flex items-center gap-2`},ql=[`disabled`],Jl=[`disabled`],Yl=[`disabled`],Xl={key:0,class:`bg-gray-900 border border-gray-800 rounded-xl p-4 md:p-5 mb-6 md:mb-8`},Zl={class:`flex flex-col md:flex-row md:items-center md:justify-between gap-4`},Ql={class:`flex-1`},$l={class:`grid grid-cols-1 sm:grid-cols-2 gap-3`},eu={class:`text-lg font-bold text-white`},tu=[`title`],nu=[`disabled`],ru={class:`grid grid-cols-1 md:grid-cols-2 gap-4 md:gap-5`},iu={class:`bg-gray-900 border border-gray-800 rounded-xl p-4 md:p-5`},au={class:`space-y-2`},ou=[`title`],su={class:`flex-1 bg-gray-800 rounded-full h-4 md:h-5 overflow-hidden`},cu={class:`w-12 md:w-16 text-xs text-gray-500 text-right shrink-0`},lu={class:`bg-gray-900 border border-gray-800 rounded-xl p-4 md:p-5`},uu={class:`space-y-2`},du={class:`w-8 md:w-10 text-xs text-gray-400 shrink-0 font-mono`},fu={class:`flex-1 bg-gray-800 rounded-full h-4 md:h-5 overflow-hidden`},pu={class:`w-12 md:w-16 text-xs text-gray-500 text-right shrink-0`},mu={__name:`Dashboard`,setup(e){let t=F(null),n=F(!0),r=F(!1),i=F(null),a=null,o=F(0),s=F(0),c=F(0),l=F(!1),u=F(null);ir(async()=>{await Promise.all([d(),f(),m(),h()]),a=setInterval(()=>{d(),f(),m(),h()},5e3)}),cr(()=>{a&&clearInterval(a)});async function d(){try{t.value=await dl(),i.value=null}catch(e){i.value=`无法加载统计数据,可能人服务器未启动或端口不对`,console.error(e)}finally{n.value=!1}}async function f(){try{let e=await gl();o.value=e.configured,s.value=e.active,c.value=e.configured}catch(e){console.error(`Failed to load workers:`,e)}}async function p(){let e=parseInt(c.value,10);if(isNaN(e)||e<1||e>500){i.value=`线程数必须在 1~500 之间`;return}l.value=!0;try{await _l(e),o.value=e,i.value=null}catch(e){i.value=`修改线程数失败: `+e.message}finally{l.value=!1}}async function m(){try{backlinkStatus.value=await vl()}catch(e){console.error(`Failed to load backlink status:`,e)}}async function h(){try{u.value=await bl()}catch(e){console.error(`Failed to load priority status:`,e)}}async function g(){backlinkTriggering.value=!0;try{await yl(),backlinkStatus.value=await vl(),i.value=null}catch(e){i.value=`触发反链计算失败: `+e.message}finally{backlinkTriggering.value=!1}}function _(e){if(!e)return`--`;let t=new Date(e),n=t-new Date,r=Math.floor(n/6e4),i=Math.floor(r/60),a=Math.floor(i/24),o=t.toLocaleTimeString(`zh-CN`,{hour:`2-digit`,minute:`2-digit`}),s=t.toLocaleDateString(`zh-CN`,{month:`short`,day:`numeric`});return a>0?`${s} ${o} (${a}天后)`:i>0?`${s} ${o} (${i}小时后)`:r>0?`${o} (${r}分钟后)`:`${o} (即将执行)`}function v(e){return e?new Date(e).toLocaleString(`zh-CN`,{month:`short`,day:`numeric`,hour:`2-digit`,minute:`2-digit`}):`从未执行`}async function y(){r.value=!0;try{await hl(),t.value=await dl()}catch(e){i.value=`刷盘失败: `+e.message}finally{r.value=!1}}function b(e){return!e&&e!==0?`0`:Number(e).toLocaleString()}function x(e){return e?Object.entries(e).sort((e,t)=>t[1]-e[1]).slice(0,10):[]}function S(e){let t={zh:`#e53e3e`,en:`#3182ce`,ja:`#e53e3e`,ko:`#3182ce`,fr:`#38a169`,de:`#d69e2e`,es:`#38a169`,ru:`#805ad5`,other:`#718096`};return t[e]||t.other}return(e,a)=>(U(),W(`div`,xl,[a[23]||=G(`h1`,{class:`text-xl md:text-2xl font-semibold text-white mb-6 md:mb-8`},`概览`,-1),n.value?(U(),W(`div`,Sl,[...a[3]||=[G(`div`,{class:`text-gray-400 animate-pulse`},`加载中...`,-1)]])):i.value?(U(),W(`div`,Cl,O(i.value),1)):t.value?(U(),W(V,{key:2},[G(`div`,wl,[G(`div`,Tl,[a[4]||=G(`div`,{class:`text-sm text-gray-500 mb-2`},`已爬取 URL`,-1),G(`div`,El,O(b(t.value.total_urls)),1)]),G(`div`,Dl,[a[5]||=G(`div`,{class:`text-sm text-gray-500 mb-2`},`总词数`,-1),G(`div`,Ol,O(b(t.value.total_words)),1)]),G(`div`,kl,[a[6]||=G(`div`,{class:`text-sm text-gray-500 mb-2`},`域名数量`,-1),G(`div`,Al,O(b(t.value.total_domains)),1)]),G(`div`,jl,[a[7]||=G(`div`,{class:`text-sm text-gray-500 mb-2`},`待重爬`,-1),G(`div`,{class:D([`text-3xl font-bold`,t.value.recrawl_eligible>0?`text-orange-400`:`text-green-400`])},O(b(t.value.recrawl_eligible)),3)]),G(`div`,Ml,[G(`div`,null,[a[8]||=G(`div`,{class:`text-sm text-gray-500 mb-2`},`待刷盘`,-1),G(`div`,{class:D([`text-3xl font-bold`,t.value.pending>0?`text-yellow-400`:`text-green-400`])},O(b(t.value.pending)),3)]),G(`button`,{class:`mt-3 w-full bg-blue-700 hover:bg-blue-600 disabled:bg-gray-700 disabled:text-gray-500 text-white text-sm font-medium py-1.5 px-3 rounded transition-colors cursor-pointer`,disabled:r.value||!t.value.pending,onClick:y},O(r.value?`刷盘中...`:`立即刷盘`),9,Nl)])]),G(`div`,Pl,[G(`div`,Fl,[G(`div`,null,[a[13]||=G(`h2`,{class:`text-sm font-semibold text-gray-300 uppercase tracking-wider mb-2`},`爬虫线程`,-1),G(`div`,Il,[G(`div`,null,[a[9]||=G(`div`,{class:`text-xs text-gray-500 mb-0.5`},`实际运行`,-1),G(`div`,{class:D([`text-3xl font-bold`,s.value>0?`text-green-400`:`text-gray-500`])},O(s.value),3)]),a[12]||=G(`div`,{class:`text-gray-700 text-xl`},`/`,-1),G(`div`,null,[a[10]||=G(`div`,{class:`text-xs text-gray-500 mb-0.5`},`设定上限`,-1),G(`div`,Ll,O(o.value),1)]),u.value&&u.value.active>0?(U(),W(`div`,Rl,[a[11]||=G(`div`,{class:`text-xs text-orange-400 mb-0.5`},`优先线程`,-1),G(`div`,zl,`+`+O(u.value.active),1)])):K(``,!0)])]),u.value?(U(),W(`div`,Bl,[G(`div`,Vl,[a[14]||=G(`div`,{class:`text-xs text-gray-500 mb-1`},`优先子链接`,-1),G(`div`,{class:D([`text-2xl font-bold`,(u.value.children_queue||0)>0?`text-yellow-400`:`text-gray-500`])},O(u.value.children_queue||0),3)]),G(`div`,Hl,[a[15]||=G(`div`,{class:`text-xs text-gray-500 mb-1`},`手动待爬`,-1),G(`div`,{class:D([`text-2xl font-bold`,u.value.pending>0?`text-yellow-400`:`text-gray-500`])},O(u.value.pending),3)]),G(`div`,Ul,[a[16]||=G(`div`,{class:`text-xs text-gray-500 mb-1`},`手动爬取中`,-1),G(`div`,{class:D([`text-2xl font-bold`,u.value.active>0?`text-orange-400`:`text-gray-500`])},O(u.value.active),3)]),G(`div`,Wl,[a[17]||=G(`div`,{class:`text-xs text-gray-500 mb-1`},`优先上限`,-1),G(`div`,Gl,O(u.value.max_workers),1)])])):K(``,!0),G(`div`,Kl,[G(`button`,{class:`w-9 h-9 flex items-center justify-center rounded-lg bg-gray-800 hover:bg-gray-700 text-gray-300 hover:text-white text-lg font-bold transition-colors cursor-pointer`,onClick:a[0]||=e=>c.value=Math.max(1,c.value-1),disabled:l.value},`−`,8,ql),An(G(`input`,{type:`number`,"onUpdate:modelValue":a[1]||=e=>c.value=e,min:`1`,max:`500`,class:`w-20 h-9 text-center bg-gray-800 border border-gray-700 rounded-lg text-white text-sm focus:outline-none focus:border-blue-500 transition-colors [appearance:textfield] [&::-webkit-outer-spin-button]:appearance-none [&::-webkit-inner-spin-button]:appearance-none`,onKeyup:fo(p,[`enter`])},null,544),[[oo,c.value,void 0,{number:!0}]]),G(`button`,{class:`w-9 h-9 flex items-center justify-center rounded-lg bg-gray-800 hover:bg-gray-700 text-gray-300 hover:text-white text-lg font-bold transition-colors cursor-pointer`,onClick:a[2]||=e=>c.value=Math.min(500,c.value+1),disabled:l.value},`+`,8,Jl),G(`button`,{class:`h-9 px-4 bg-blue-700 hover:bg-blue-600 disabled:bg-gray-700 disabled:text-gray-500 text-white text-sm font-medium rounded-lg transition-colors cursor-pointer`,disabled:l.value||c.value===o.value,onClick:p},O(l.value?`保存中...`:`应用`),9,Yl)])])]),e.backlinkStatus?(U(),W(`div`,Xl,[G(`div`,Zl,[G(`div`,Ql,[a[20]||=G(`h2`,{class:`text-sm font-semibold text-gray-300 uppercase tracking-wider mb-3`},`反链计算(PageRank)`,-1),G(`div`,$l,[G(`div`,null,[a[18]||=G(`div`,{class:`text-xs text-gray-500 mb-0.5`},`下次自动执行`,-1),G(`div`,{class:D([`text-lg font-bold`,e.backlinkStatus.running?`text-yellow-400`:`text-white`])},O(e.backlinkStatus.running?`计算中...`:_(e.backlinkStatus.next_run)),3)]),G(`div`,null,[a[19]||=G(`div`,{class:`text-xs text-gray-500 mb-0.5`},`上次完成`,-1),G(`div`,eu,O(v(e.backlinkStatus.last_run)),1)])]),e.backlinkStatus.last_error?(U(),W(`div`,{key:0,class:`mt-2 text-xs text-red-400 truncate`,title:e.backlinkStatus.last_error},` 上次错误:`+O(e.backlinkStatus.last_error),9,tu)):K(``,!0)]),G(`button`,{class:`h-9 px-4 bg-blue-700 hover:bg-blue-600 disabled:bg-gray-700 disabled:text-gray-500 text-white text-sm font-medium rounded-lg transition-colors cursor-pointer whitespace-nowrap`,disabled:e.backlinkTriggering||e.backlinkStatus.running,onClick:g},O(e.backlinkTriggering?`已触发...`:e.backlinkStatus.running?`计算中...`:`立即执行`),9,nu)])])):K(``,!0),G(`div`,ru,[G(`div`,iu,[a[21]||=G(`h2`,{class:`text-sm font-semibold text-gray-300 mb-3 md:mb-4 uppercase tracking-wider`},`域名分布 Top 10`,-1),G(`div`,au,[(U(!0),W(V,null,R(x(t.value.domains),([e,n])=>(U(),W(`div`,{key:e,class:`flex items-center gap-2 md:gap-3`},[G(`div`,{class:`w-24 md:w-36 text-xs text-gray-400 truncate shrink-0`,title:e},O(e),9,ou),G(`div`,su,[G(`div`,{class:`h-full bg-blue-600 rounded-full transition-all duration-500`,style:me({width:`${n/t.value.domains[Object.keys(t.value.domains)[0]]*100}%`})},null,4)]),G(`div`,cu,O(b(n)),1)]))),128))])]),G(`div`,lu,[a[22]||=G(`h2`,{class:`text-sm font-semibold text-gray-300 mb-3 md:mb-4 uppercase tracking-wider`},`语种分布`,-1),G(`div`,uu,[(U(!0),W(V,null,R(Object.entries(t.value.languages||{}).sort((e,t)=>t[1]-e[1]),([e,n])=>(U(),W(`div`,{key:e,class:`flex items-center gap-2 md:gap-3`},[G(`div`,du,O(e),1),G(`div`,fu,[G(`div`,{class:`h-full rounded-full transition-all duration-500`,style:me({width:`${n/t.value.total_urls*100}%`,backgroundColor:S(e)})},null,4)]),G(`div`,pu,O(b(n)),1)]))),128))])])])],64)):K(``,!0)]))}},hu={class:`p-4 md:p-8`},gu={class:`flex flex-col md:flex-row md:items-center justify-between mb-4 md:mb-6 gap-3`},_u={class:`text-sm text-gray-500`},vu={class:`flex items-center gap-2 md:gap-3`},yu=[`value`],bu={class:`flex flex-col sm:flex-row items-stretch sm:items-center gap-3 mb-4 md:mb-5`},xu={class:`relative flex-1 max-w-full sm:max-w-sm`},Su={key:0,class:`flex items-center justify-center h-48`},Cu={key:1,class:`bg-red-900/30 border border-red-800 rounded-lg p-4 text-red-300`},wu={key:2,class:`bg-gray-900 border border-gray-800 rounded-xl overflow-hidden`},Tu={class:`hidden md:table w-full text-sm`},Eu={class:`px-5 py-3.5`},Du=[`href`],Ou={class:`font-medium text-gray-200 group-hover:text-white line-clamp-2`},ku={class:`text-xs text-gray-600 mt-0.5 break-all line-clamp-1`},Au={key:0,class:`text-xs text-gray-500 mt-1 line-clamp-1`},ju={class:`px-5 py-3.5`},Mu={class:`text-gray-400 text-xs font-mono`},Nu={class:`px-5 py-3.5`},Pu={key:1,class:`text-xs text-gray-600`},Fu={class:`px-5 py-3.5 text-gray-500 text-xs tabular-nums`},Iu={class:`px-5 py-3.5 text-gray-500 text-xs`},Lu={key:0},Ru={class:`md:hidden divide-y divide-gray-800`},zu=[`href`],Bu={class:`font-medium text-gray-200 text-sm line-clamp-2 mb-1`},Vu={class:`text-xs text-gray-500 break-all line-clamp-1 mb-2`},Hu={class:`flex items-center gap-2 text-xs`},Uu={class:`text-gray-400 font-mono`},Wu={class:`text-gray-600 ml-auto`},Gu={key:0,class:`p-8 text-center text-gray-600`},Ku={key:3,class:`mt-3 text-xs text-gray-600 text-right pb-4 md:pb-0`},qu={__name:`RecentCrawls`,setup(e){let t=F([]),n=F(0),r=F(!0),i=F(null),a=F(``),o=F(``),s=[20,50,100,200],c=F(50);ir(async()=>{await l()});async function l(){r.value=!0,i.value=null;try{let e=await ul(c.value);t.value=e.items||[],n.value=e.total||0}catch(e){i.value=`无法加载数据,可能人服务器未启动`,console.error(e)}finally{r.value=!1}}async function u(e){c.value=e,await l()}let d=ha(()=>{let e=t.value;if(a.value){let t=a.value.toLowerCase();e=e.filter(e=>e.title?.toLowerCase().includes(t)||e.url?.toLowerCase().includes(t)||e.domain?.toLowerCase().includes(t))}return o.value&&(e=e.filter(e=>Object.keys(e.language||{}).includes(o.value))),e});function f(e){return e?new Date(e*1e3).toLocaleString(`zh-CN`,{year:`numeric`,month:`2-digit`,day:`2-digit`,hour:`2-digit`,minute:`2-digit`,second:`2-digit`,hour12:!1}):`-`}function p(e){return{zh:{label:`中文`,cls:`bg-red-900/60 text-red-300`},en:{label:`EN`,cls:`bg-blue-900/60 text-blue-300`},ja:{label:`日`,cls:`bg-pink-900/60 text-pink-300`},ko:{label:`한`,cls:`bg-blue-900/60 text-blue-300`},fr:{label:`FR`,cls:`bg-green-900/60 text-green-300`},de:{label:`DE`,cls:`bg-yellow-900/60 text-yellow-300`},es:{label:`ES`,cls:`bg-green-900/60 text-green-300`},ru:{label:`RU`,cls:`bg-purple-900/60 text-purple-300`}}[e]||{label:e,cls:`bg-gray-800 text-gray-400`}}function m(e){return e?Object.entries(e).sort((e,t)=>t[1]-e[1])[0]:null}return(e,t)=>(U(),W(`div`,hu,[G(`div`,gu,[G(`div`,null,[t[4]||=G(`h1`,{class:`text-xl md:text-2xl font-semibold text-white mb-1`},`最近爬取`,-1),G(`p`,_u,`共 `+O(n.value.toLocaleString())+` 条记录`,1)]),G(`div`,vu,[An(G(`select`,{"onUpdate:modelValue":t[0]||=e=>c.value=e,onChange:t[1]||=e=>u(c.value),class:`bg-gray-900 border border-gray-700 text-gray-300 text-sm rounded-lg px-3 py-2 focus:border-blue-500 focus:outline-none`},[(U(),W(V,null,R(s,e=>G(`option`,{key:e,value:e},`显示 `+O(e)+` 条`,9,yu)),64))],544),[[so,c.value]]),G(`button`,{onClick:l,class:`bg-blue-600 hover:bg-blue-700 text-white text-sm px-4 py-2 rounded-lg transition-colors`},` 刷新 `)])]),G(`div`,bu,[G(`div`,xu,[An(G(`input`,{"onUpdate:modelValue":t[2]||=e=>a.value=e,type:`text`,placeholder:`搜索标题、URL、域名...`,class:`w-full bg-gray-900 border border-gray-700 text-gray-200 text-sm rounded-lg pl-10 pr-4 py-2 focus:border-blue-500 focus:outline-none placeholder-gray-600`},null,512),[[oo,a.value]]),t[5]||=G(`span`,{class:`absolute left-3 top-1/2 -translate-y-1/2 text-gray-500`},`🔍`,-1)]),An(G(`select`,{"onUpdate:modelValue":t[3]||=e=>o.value=e,class:`bg-gray-900 border border-gray-700 text-gray-300 text-sm rounded-lg px-3 py-2 focus:border-blue-500 focus:outline-none`},[...t[6]||=[Ui(``,9)]],512),[[so,o.value]])]),r.value?(U(),W(`div`,Su,[...t[7]||=[G(`div`,{class:`text-gray-400 animate-pulse`},`加载中...`,-1)]])):i.value?(U(),W(`div`,Cu,O(i.value),1)):(U(),W(`div`,wu,[G(`table`,Tu,[t[9]||=G(`thead`,null,[G(`tr`,{class:`border-b border-gray-800`},[G(`th`,{class:`text-left px-5 py-3 text-gray-500 font-medium text-xs uppercase tracking-wider`},`标题`),G(`th`,{class:`text-left px-5 py-3 text-gray-500 font-medium text-xs uppercase tracking-wider w-28`},`域名`),G(`th`,{class:`text-left px-5 py-3 text-gray-500 font-medium text-xs uppercase tracking-wider w-16`},`语种`),G(`th`,{class:`text-left px-5 py-3 text-gray-500 font-medium text-xs uppercase tracking-wider w-20`},`字数`),G(`th`,{class:`text-left px-5 py-3 text-gray-500 font-medium text-xs uppercase tracking-wider w-48`},`爬取时间`)])],-1),G(`tbody`,null,[(U(!0),W(V,null,R(d.value,e=>(U(),W(`tr`,{key:e.url,class:`border-b border-gray-800/50 hover:bg-gray-800/40 transition-colors group`},[G(`td`,Eu,[G(`a`,{href:e.url,target:`_blank`,rel:`noopener noreferrer`,class:`block hover:opacity-80 transition-opacity`},[G(`div`,Ou,O(e.title||`(无标题)`),1),G(`div`,ku,O(e.url),1)],8,Du),e.description?(U(),W(`div`,Au,O(e.description),1)):K(``,!0)]),G(`td`,ju,[G(`span`,Mu,O(e.domain),1)]),G(`td`,Nu,[m(e.language)?(U(),W(`span`,{key:0,class:D([`text-xs px-1.5 py-0.5 rounded font-medium`,p(m(e.language)[0]).cls])},O(p(m(e.language)[0]).label)+` `+O((m(e.language)[1]*100).toFixed(0))+`% `,3)):(U(),W(`span`,Pu,`-`))]),G(`td`,Fu,O(e.word_count.toLocaleString()),1),G(`td`,Iu,O(f(e.crawled_at)),1)]))),128)),d.value.length?K(``,!0):(U(),W(`tr`,Lu,[...t[8]||=[G(`td`,{colspan:`5`,class:`px-5 py-12 text-center text-gray-600`},` 没有找到匹配的记录 `,-1)]]))])]),G(`div`,Ru,[(U(!0),W(V,null,R(d.value,e=>(U(),W(`a`,{key:e.url,href:e.url,target:`_blank`,rel:`noopener noreferrer`,class:`block p-4 hover:bg-gray-800/40 transition-colors`},[G(`div`,Bu,O(e.title||`(无标题)`),1),G(`div`,Vu,O(e.url),1),G(`div`,Hu,[G(`span`,Uu,O(e.domain),1),m(e.language)?(U(),W(`span`,{key:0,class:D([`px-1.5 py-0.5 rounded font-medium`,p(m(e.language)[0]).cls])},O(p(m(e.language)[0]).label),3)):K(``,!0),G(`span`,Wu,O(f(e.crawled_at)),1)])],8,zu))),128)),d.value.length?K(``,!0):(U(),W(`div`,Gu,` 没有找到匹配的记录 `))])])),!r.value&&!i.value&&d.value.length?(U(),W(`div`,Ku,` 筛选后 `+O(d.value.length)+` 条 / 共 `+O(n.value.toLocaleString())+` 条 `,1)):K(``,!0)]))}},Ju={class:`p-4 md:p-8`},Yu={class:`bg-gray-900 rounded-xl p-4 md:p-6 mb-4 md:mb-6 border border-gray-800`},Xu={class:`flex flex-col sm:flex-row gap-3`},Zu=[`disabled`],Qu=[`disabled`],$u={key:0,class:`mt-3 text-sm text-red-400`},ed={key:1,class:`mt-3 text-sm text-green-400`},td={class:`bg-gray-900 rounded-xl border border-gray-800`},nd={class:`px-4 md:px-6 py-3 md:py-4 border-b border-gray-800 flex items-center justify-between`},rd={class:`text-xs text-gray-500`},id={key:0,class:`p-8 text-center text-gray-500 text-sm`},ad={key:1,class:`p-8 text-center text-red-400 text-sm`},od={key:2,class:`p-8 text-center text-gray-500 text-sm`},sd={key:3,class:`hidden md:table w-full text-sm`},cd={class:`divide-y divide-gray-800`},ld={class:`px-6 py-3`},ud={class:`text-gray-300 break-all`},dd={class:`px-6 py-3`},fd={key:0,class:`inline-block px-2 py-0.5 text-xs rounded bg-purple-900 text-purple-300`},pd={key:1,class:`inline-block px-2 py-0.5 text-xs rounded bg-blue-900 text-blue-300`},md={class:`px-6 py-3 text-gray-500`},hd={class:`px-6 py-3`},gd=[`onClick`],_d={class:`md:hidden divide-y divide-gray-800`},vd={class:`flex-1 min-w-0`},yd={class:`text-gray-300 text-sm break-all mb-1`},bd={class:`flex items-center gap-2`},xd={key:0,class:`inline-block px-2 py-0.5 text-xs rounded bg-purple-900 text-purple-300`},Sd={key:1,class:`inline-block px-2 py-0.5 text-xs rounded bg-blue-900 text-blue-300`},Cd={class:`text-xs text-gray-500`},wd=[`onClick`],Td={__name:`PriorityCrawl`,setup(e){let t=F([]),n=F(!0),r=F(null),i=F(!1),a=F(null),o=F(!1),s=F(``);async function c(){n.value=!0,r.value=null;try{t.value=(await fl()).items||[]}catch(e){r.value=`加载失败,请检查人服务器是否启动`,console.error(e)}finally{n.value=!1}}async function l(){let e=s.value.trim();if(e){i.value=!0,a.value=null,o.value=!1;try{await pl(e),s.value=``,o.value=!0,setTimeout(()=>{o.value=!1},3e3),await c()}catch(e){a.value=e?.response?.data?.error||`添加失败`}finally{i.value=!1}}}async function u(e){try{await ml(e),await c()}catch(e){r.value=`删除失败`,console.error(e)}}function d(e){return e?new Date(e*1e3).toLocaleString(`zh-CN`,{month:`2-digit`,day:`2-digit`,hour:`2-digit`,minute:`2-digit`}):`-`}return ir(c),(e,c)=>(U(),W(`div`,Ju,[c[3]||=G(`h1`,{class:`text-xl md:text-2xl font-semibold text-white mb-2`},`插入爬取`,-1),c[4]||=G(`p`,{class:`text-sm text-gray-500 mb-6 md:mb-8`},` 添加 URL 或域名,下一轮爬取时会优先抓取。纯域名会自动补全为 https://www.域名/。 `,-1),G(`div`,Yu,[G(`div`,Xu,[An(G(`input`,{"onUpdate:modelValue":c[0]||=e=>s.value=e,onKeyup:fo(l,[`enter`]),type:`text`,placeholder:`输入 URL 或域名,例如 https://example.com 或 example.com`,class:`flex-1 bg-gray-800 border border-gray-700 rounded-lg px-4 py-2.5 text-gray-100 placeholder-gray-500 focus:outline-none focus:border-blue-500 focus:ring-1 focus:ring-blue-500 text-sm`,disabled:i.value},null,40,Zu),[[oo,s.value]]),G(`button`,{onClick:l,disabled:i.value||!s.value.trim(),class:`px-6 py-2.5 bg-blue-600 hover:bg-blue-500 disabled:bg-gray-700 disabled:text-gray-500 text-white text-sm font-medium rounded-lg transition-colors cursor-pointer whitespace-nowrap`},O(i.value?`添加中...`:`插入队列`),9,Qu)]),a.value?(U(),W(`div`,$u,O(a.value),1)):K(``,!0),o.value?(U(),W(`div`,ed,` 已添加到优先队列,将在下一轮爬取时优先抓取 `)):K(``,!0)]),G(`div`,td,[G(`div`,nd,[c[1]||=G(`span`,{class:`text-sm font-medium text-gray-300`},`待爬取队列`,-1),G(`span`,rd,O(t.value.length)+` 条`,1)]),n.value?(U(),W(`div`,id,` 加载中... `)):r.value?(U(),W(`div`,ad,O(r.value),1)):t.value.length===0?(U(),W(`div`,od,` 暂无待爬取的优先 URL `)):(U(),W(`table`,sd,[c[2]||=G(`thead`,null,[G(`tr`,{class:`text-left text-gray-500 text-xs border-b border-gray-800`},[G(`th`,{class:`px-6 py-3 font-medium`},`URL`),G(`th`,{class:`px-6 py-3 font-medium w-28`},`类型`),G(`th`,{class:`px-6 py-3 font-medium w-40`},`添加时间`),G(`th`,{class:`px-6 py-3 font-medium w-16`},`操作`)])],-1),G(`tbody`,cd,[(U(!0),W(V,null,R(t.value,e=>(U(),W(`tr`,{key:e.url,class:`hover:bg-gray-800/50`},[G(`td`,ld,[G(`span`,ud,O(e.url),1)]),G(`td`,dd,[e.domain?(U(),W(`span`,fd,`域名`)):(U(),W(`span`,pd,`URL`))]),G(`td`,md,O(d(e.added_at)),1),G(`td`,hd,[G(`button`,{onClick:t=>u(e.url),class:`text-red-400 hover:text-red-300 text-xs cursor-pointer`},` 删除 `,8,gd)])]))),128))])])),G(`div`,_d,[(U(!0),W(V,null,R(t.value,e=>(U(),W(`div`,{key:e.url,class:`p-4 hover:bg-gray-800/50 flex items-center justify-between gap-3`},[G(`div`,vd,[G(`div`,yd,O(e.url),1),G(`div`,bd,[e.domain?(U(),W(`span`,xd,`域名`)):(U(),W(`span`,Sd,`URL`)),G(`span`,Cd,O(d(e.added_at)),1)])]),G(`button`,{onClick:t=>u(e.url),class:`text-red-400 hover:text-red-300 text-xs cursor-pointer shrink-0 px-2 py-1`},` 删除 `,8,wd)]))),128))])])]))}},Ed={class:`flex flex-col h-full`},Dd={class:`bg-gray-950 border-b border-gray-800 px-4 md:px-8 py-4 md:py-6`},Od={class:`max-w-3xl mx-auto`},kd={class:`relative`},Ad={class:`flex-1 overflow-y-auto px-8 py-6`},jd={class:`max-w-3xl mx-auto`},Md={key:0,class:`flex items-center gap-3 text-gray-400 py-8`},Nd={key:1,class:`bg-red-900/30 border border-red-800 rounded-lg p-4 text-red-300`},Pd={key:2,class:`py-16 text-center text-gray-600`},Fd={key:3,class:`py-16 text-center text-gray-600`},Id={key:4,class:`flex flex-col sm:flex-row sm:items-center gap-2 sm:gap-4 mb-4 md:mb-5 text-sm text-gray-500`},Ld={class:`flex items-center gap-2`},Rd={class:`text-gray-300`},zd={class:`flex flex-wrap gap-2 sm:ml-auto`},Bd={class:`text-gray-300`},Vd={key:5,class:`space-y-2 md:space-y-1`},Hd=[`onClick`],Ud={class:`flex items-start gap-2 md:gap-3 mb-2`},Wd={class:`flex-1 min-w-0`},Gd={class:`text-blue-400 group-hover:text-blue-300 text-base md:text-lg leading-snug`},Kd={class:`text-xs text-gray-600 mt-0.5 truncate`},qd={class:`flex flex-col items-end gap-1 shrink-0`},Jd={class:`text-xs text-gray-600`},Yd={class:`w-12 md:w-14 bg-gray-800 rounded-full h-1.5 overflow-hidden`},Xd={class:`text-gray-400 text-sm leading-relaxed mb-2 md:mb-3`},Zd={class:`flex flex-wrap items-center gap-2 md:gap-3 text-xs`},Qd={class:`flex flex-wrap gap-1.5`},$d=[`title`],ef={class:`text-blue-400`},tf={class:`text-gray-600 ml-1`},nf={key:0,class:`text-gray-600`},rf={class:`text-gray-700 ml-0 md:ml-auto`},af={class:`text-xs text-gray-700 mt-1 truncate`},of={key:6,class:`mt-6 text-center`},sf=``,cf=10,lf={__name:`SearchView`,setup(e){let t=F(``),n=F([]),r=F(0),i=F({}),a=F(!1),o=F(null),s=F(0),c=null;async function l(e,t=!1){if(!e.trim()){n.value=[],r.value=0,i.value={};return}t||(s.value=0),a.value=!0,o.value=null;try{let a=encodeURIComponent(e),o=s.value*cf,c=await fetch(`${sf}/search?qh=${a}&slice=${o}:${o+cf}`);if(!c.ok)throw Error(`HTTP ${c.status}`);let l=await c.json();t?n.value=[...n.value,...l.results||[]]:n.value=l.results||[],r.value=l.total||0,i.value=l.counts||{}}catch(e){o.value=`搜索失败,可能是人服务器未启动`,console.error(e)}finally{a.value=!1}}function u(){clearTimeout(c),c=setTimeout(()=>l(t.value),400)}function d(e){e.key===`Enter`&&(clearTimeout(c),l(t.value))}function f(e,t=120){return e?e.length>t?e.slice(0,t)+`…`:e:``}function p(e){return e?e.toLocaleString():`0`}function m(e){return Math.min(100,Math.round(e*100))}function h(e){return{zh:`中文`,en:`EN`,ja:`JA`,ko:`KO`,fr:`FR`,de:`DE`,es:`ES`,ru:`RU`}[e]||e?.toUpperCase()}function g(e){try{return new URL(e).hostname}catch{return e}}function _(e){window.open(e,`_blank`)}return(e,c)=>(U(),W(`div`,Ed,[G(`div`,Dd,[G(`div`,Od,[G(`div`,kd,[An(G(`input`,{"onUpdate:modelValue":c[0]||=e=>t.value=e,onInput:u,onKeydown:d,type:`text`,placeholder:`输入关键词搜索,或用 site:example.com 限定域名`,class:`w-full bg-gray-900 border border-gray-700 rounded-xl md:rounded-2xl px-4 md:px-6 py-3 md:py-4 pr-20 md:pr-14 text-white text-base md:text-lg placeholder-gray-600 focus:outline-none focus:border-blue-500 focus:ring-1 focus:ring-blue-500 transition`},null,544),[[oo,t.value]]),G(`button`,{onClick:c[1]||=e=>l(t.value),class:`absolute right-2 md:right-3 top-1/2 -translate-y-1/2 bg-blue-600 hover:bg-blue-500 text-white rounded-lg md:rounded-xl px-3 md:px-4 py-1.5 md:py-2 text-sm font-medium transition`},` 搜索 `)])])]),G(`div`,Ad,[G(`div`,jd,[a.value?(U(),W(`div`,Md,[...c[3]||=[G(`div`,{class:`w-5 h-5 border-2 border-blue-400 border-t-transparent rounded-full animate-spin`},null,-1),G(`span`,null,`搜索中...`,-1)]])):o.value?(U(),W(`div`,Nd,O(o.value),1)):t.value.trim()?n.value.length===0&&!a.value?(U(),W(`div`,Fd,` 未找到相关结果 `)):n.value.length>0?(U(),W(`div`,Id,[G(`div`,Ld,[G(`span`,null,[c[4]||=Hi(`找到约 `,-1),G(`strong`,Rd,O(p(r.value)),1),c[5]||=Hi(` 条结果`,-1)]),c[6]||=G(`span`,{class:`text-gray-700`},`|`,-1),G(`span`,null,O(n.value.length)+` 条已加载`,1)]),G(`div`,zd,[(U(!0),W(V,null,R(i.value,(e,t)=>(U(),W(`span`,{key:t,class:`inline-flex items-center gap-1 bg-gray-800 rounded px-2 py-0.5 text-xs text-gray-400`},[G(`span`,Bd,O(t),1),G(`span`,null,O(p(e)),1)]))),128))])])):K(``,!0):(U(),W(`div`,Pd,` 输入关键词开始搜索 `)),n.value.length>0?(U(),W(`div`,Vd,[(U(!0),W(V,null,R(n.value,(e,t)=>(U(),W(`div`,{key:t,onClick:t=>_(e.url),class:`group block bg-gray-900/50 hover:bg-gray-900 border border-gray-800 hover:border-gray-700 rounded-xl p-4 md:p-5 cursor-pointer transition`},[G(`div`,Ud,[G(`div`,Wd,[G(`div`,Gd,O(e.snippet?.title||g(e.url)),1),G(`div`,Kd,O(g(e.url)),1)]),G(`div`,qd,[G(`div`,Jd,O(m(e.score))+`%`,1),G(`div`,Yd,[G(`div`,{class:`h-full bg-blue-500 rounded-full`,style:me({width:m(e.score)+`%`})},null,4)])])]),G(`p`,Xd,O(f(e.snippet?.description||e.snippet?.text)),1),G(`div`,Zd,[G(`div`,Qd,[(U(!0),W(V,null,R(e.relevance,(e,t)=>(U(),W(`span`,{key:t,class:`inline-flex items-center bg-gray-800 rounded px-1.5 py-0.5`,title:`${t}: ${(e*100).toFixed(1)}%`},[G(`span`,ef,O(t),1),G(`span`,tf,O((e*100).toFixed(0))+`%`,1)],8,$d))),128))]),e.snippet?.language?(U(),W(`span`,nf,O(h(e.snippet.language)),1)):K(``,!0),G(`span`,rf,O(p(e.domain_count))+` 个结果`,1)]),G(`div`,af,O(e.url),1)],8,Hd))),128))])):K(``,!0),n.value.length>0&&n.value.length{s.value++,l(t.value,!0)},class:`bg-gray-800 hover:bg-gray-700 text-gray-300 rounded-xl px-6 py-2.5 text-sm transition`},` 加载更多 (`+O(n.value.length)+`/`+O(p(r.value))+`) `,1)])):K(``,!0)])])]))}},uf={class:`flex h-screen bg-gray-950 text-gray-100 font-sans`},df={class:`hidden md:flex w-56 bg-gray-900 border-r border-gray-800 flex-col shrink-0`},ff={class:`flex-1 py-4 px-3 space-y-1`},pf=[`onClick`],mf={class:`flex-1 overflow-y-auto pb-16 md:pb-0`},hf={class:`md:hidden fixed bottom-0 left-0 right-0 bg-gray-900 border-t border-gray-800 flex justify-around items-center h-16 z-50`},gf=[`onClick`],_f={class:`text-lg`},vf={class:`text-[10px]`};go({__name:`App`,setup(e){let t=F(`dashboard`),n=[{id:`dashboard`,label:`概览`,icon:`📊`},{id:`recent`,label:`最近`,icon:`🕷️`},{id:`search`,label:`搜索`,icon:`🔍`},{id:`priority`,label:`插入`,icon:`🚀`}];return(e,r)=>(U(),W(`div`,uf,[G(`aside`,df,[r[0]||=G(`div`,{class:`px-5 py-5 border-b border-gray-800`},[G(`div`,{class:`text-lg font-semibold text-white tracking-tight`},`SESE Admin`),G(`div`,{class:`text-xs text-gray-500 mt-0.5`},`爬取内容监控`)],-1),G(`nav`,ff,[(U(),W(V,null,R(n,e=>G(`button`,{key:e.id,onClick:n=>t.value=e.id,class:D([`w-full flex items-center gap-3 px-3 py-2.5 rounded-lg text-sm font-medium transition-colors`,t.value===e.id?`bg-blue-600 text-white`:`text-gray-400 hover:text-white hover:bg-gray-800`])},[G(`span`,null,O(e.icon),1),Hi(` `+O(e.label),1)],10,pf)),64))]),r[1]||=G(`div`,{class:`px-5 py-4 border-t border-gray-800`},[G(`div`,{class:`text-xs text-gray-600`},`sese-engine v1.0`)],-1)]),G(`main`,mf,[t.value===`dashboard`?(U(),Ni(mu,{key:0})):t.value===`recent`?(U(),Ni(qu,{key:1})):t.value===`search`?(U(),Ni(lf,{key:2})):t.value===`priority`?(U(),Ni(Td,{key:3})):K(``,!0)]),G(`nav`,hf,[(U(),W(V,null,R(n,e=>G(`button`,{key:e.id,onClick:n=>t.value=e.id,class:D([`flex flex-col items-center justify-center gap-0.5 flex-1 h-full transition-colors`,t.value===e.id?`text-blue-400`:`text-gray-500`])},[G(`span`,_f,O(e.icon),1),G(`span`,vf,O(e.label),1)],10,gf)),64))])]))}}).mount(`#app`); \ No newline at end of file diff --git a/dist/index.html b/dist/index.html index c787826..1a0632c 100644 --- a/dist/index.html +++ b/dist/index.html @@ -5,7 +5,7 @@ SESE 爬取管理 - +