추가된 도서관 시스템: • 광주남구도서관 (5개관) • 광주시교육청통합도서관 (6개관) • 전남교육청통합도서관 (25개관) • 전남교육청행정자료실 (1개관) • 여수시립도서관 (34개관) • 고흥군립도서관 (7개관) • 광주북구통합도서관 (3개관) • 광주북구작은도서관 (23개관) • 광주북구공공도서관 (5개관) • 전북교육청도서관 (18개관) • 광주광산구통합도서관 (17개관) • 목포시립도서관 (23개관) • 순천시립도서관 (10개관) • 광주시립도서관 (4개관) • 완도군립도서관 (6개관) • 익산시통합도서관 (33개관) • 안산시중앙도서관 (27개관) • 광주서구구립도서관 (4개관) • 광주서구스마트도서관 (4개관) • 광주서구작은도서관 (5개관) • 광주동구도서관 (5개관) • 경남대표도서관 (1개관) • 무안군립도서관 (1개관) • 조선대학교중앙도서관 (1개관) • 조선이공대학교도서관 (1개관) • KCM통합도서관 (33개관) 총 400개 이상 도서관 복본조사 시스템 완성 HTTP API 방식 및 Selenium 크롤링 방식 혼용 브라우저 헤더 최적화 및 80% 화면배율 적용 🤖 Generated with [Claude Code](https://claude.ai/code) Co-Authored-By: Claude <noreply@anthropic.com>
644 lines
26 KiB
C#
644 lines
26 KiB
C#
using AngleSharp.Dom;
|
|
using AR;
|
|
using OpenQA.Selenium;
|
|
using OpenQA.Selenium.Chrome;
|
|
using OpenQA.Selenium.Chromium;
|
|
using OpenQA.Selenium.Interactions;
|
|
using OpenQA.Selenium.Support.UI;
|
|
using System;
|
|
using System.IO;
|
|
using System.Linq;
|
|
using System.Net.Http.Headers;
|
|
using System.Runtime.CompilerServices;
|
|
using System.Runtime.InteropServices;
|
|
using System.Text.RegularExpressions;
|
|
using System.Threading;
|
|
using System.Threading.Tasks;
|
|
using System.Web.UI.WebControls;
|
|
using System.Xml.Linq;
|
|
using UniMarc.SearchModel;
|
|
using UniMarc.마크;
|
|
using WebDriverManager;
|
|
using WebDriverManager.DriverConfigs.Impl;
|
|
|
|
namespace BokBonCheck
|
|
{
|
|
public class AnsanLibSearcher : ILibrarySearcher
|
|
{
|
|
public string AreaCode { get; set; } = string.Empty;
|
|
public string SiteName { get; protected set; }
|
|
public string SiteUrl => "https://lib.ansan.go.kr/DetailSearch";
|
|
public bool HttpApiMode { get; set; } = false;
|
|
|
|
public int No { get; set; }
|
|
|
|
private ChromiumDriver _driver;
|
|
|
|
public AnsanLibSearcher(int no, string areaCode, string areaName)
|
|
{
|
|
this.No = no;
|
|
this.AreaCode = areaCode;
|
|
this.SiteName = $"안산시립({areaName})";
|
|
}
|
|
|
|
public void StopDriver()
|
|
{
|
|
if (_driver != null)
|
|
{
|
|
_driver.Quit();
|
|
_driver.Dispose();
|
|
_driver = null;
|
|
}
|
|
}
|
|
|
|
public async Task StartDriver(bool showdriver = false)
|
|
{
|
|
if (_driver == null)
|
|
{
|
|
try
|
|
{
|
|
if (SeleniumHelper.IsReady == false) await SeleniumHelper.Download();
|
|
_driver = await SeleniumHelper.CreateDriver(ShowBrowser: showdriver);
|
|
|
|
// 브라우저 배율을 80%로 설정 (브라우저가 표시되는 경우에만)
|
|
if (showdriver)
|
|
{
|
|
try
|
|
{
|
|
((IJavaScriptExecutor)_driver).ExecuteScript("document.body.style.zoom = '0.8';");
|
|
Console.WriteLine("브라우저 배율을 80%로 설정했습니다.");
|
|
}
|
|
catch (Exception zoomEx)
|
|
{
|
|
Console.WriteLine($"브라우저 배율 설정 실패: {zoomEx.Message}");
|
|
}
|
|
}
|
|
|
|
Console.WriteLine("AnsanLibSearcher Driver 초기화 완료");
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
Console.WriteLine($"AnsanLibSearcher Driver 초기화 실패: {ex.Message}");
|
|
throw new InvalidOperationException($"AnsanLibSearcher Driver 초기화에 실패했습니다: {ex.Message}", ex);
|
|
}
|
|
}
|
|
}
|
|
|
|
virtual protected bool SelectLibrary(WebDriverWait wait)
|
|
{
|
|
try
|
|
{
|
|
Console.WriteLine("도서관 선택 과정 시작...");
|
|
|
|
// 1. 열기버튼 찾기 (작은도서관 체크박스가 보이도록)
|
|
var element = wait.Until(d => d.FindElement(By.CssSelector(".btIco.plus")));
|
|
if (element != null)
|
|
{
|
|
SafeClick(element);
|
|
}
|
|
|
|
// 2. 현재 체크된 상태 확인
|
|
var allCheckboxes = wait.Until(d => d.FindElements(By.CssSelector("div.library input[type='checkbox']")));
|
|
var checkedCheckboxes = allCheckboxes.Where(cb => cb.Selected).ToList();
|
|
|
|
Console.WriteLine($"현재 체크된 체크박스 개수: {checkedCheckboxes.Count}");
|
|
|
|
// 3. 지정한 도서관만 정확히 1개 체크되어 있는지 확인
|
|
bool isTargetOnlyChecked = false;
|
|
if (!string.IsNullOrEmpty(AreaCode))
|
|
{
|
|
if (checkedCheckboxes.Count == 1)
|
|
{
|
|
var onlyChecked = checkedCheckboxes[0];
|
|
var onlyCheckedValue = onlyChecked.GetAttribute("value");
|
|
if (onlyCheckedValue == AreaCode)
|
|
{
|
|
Console.WriteLine($"✓ {AreaCode} 도서관만 정확히 선택되어 있음 - 완료");
|
|
isTargetOnlyChecked = true;
|
|
}
|
|
}
|
|
}
|
|
|
|
// 4. 목표 상태가 아니면 전체선택 버튼으로 모두 해제
|
|
if (!isTargetOnlyChecked)
|
|
{
|
|
Console.WriteLine("전체선택 버튼을 이용해서 모든 체크박스 해제 중...");
|
|
|
|
// 전체선택 버튼들 찾기 (총 2개)
|
|
var allSelectButtons = wait.Until(d => d.FindElements(By.CssSelector("span.btnAllSelect")));
|
|
Console.WriteLine($"전체선택 버튼 {allSelectButtons.Count}개 발견");
|
|
|
|
// 모든 전체선택 버튼 클릭해서 해제
|
|
foreach (var button in allSelectButtons)
|
|
{
|
|
try
|
|
{
|
|
Console.WriteLine("전체선택 버튼 클릭 중...");
|
|
SafeClick(button);
|
|
Thread.Sleep(50);
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
Console.WriteLine($"전체선택 버튼 클릭 오류: {ex.Message}");
|
|
}
|
|
}
|
|
|
|
// 해제 후 체크된 체크박스 개수 확인
|
|
var remainingChecked = allCheckboxes.Where(cb => cb.Selected).Count();
|
|
Console.WriteLine($"전체선택 버튼 클릭 후 체크된 체크박스 개수: {remainingChecked}");
|
|
|
|
// 5. 지정 도서관만 체크
|
|
if (!string.IsNullOrEmpty(AreaCode))
|
|
{
|
|
Console.WriteLine($"목표 도서관 '{AreaCode}' 선택 중...");
|
|
|
|
try
|
|
{
|
|
var targetCheckbox = wait.Until(d => d.FindElement(By.CssSelector($"div.library input[type='checkbox'][value='{AreaCode}']")));
|
|
|
|
if (!targetCheckbox.Selected)
|
|
{
|
|
Console.WriteLine($"{AreaCode} 도서관 체크 중...");
|
|
SafeClick(targetCheckbox);
|
|
Thread.Sleep(300);
|
|
}
|
|
|
|
// 최종 선택 상태 확인
|
|
if (targetCheckbox.Selected)
|
|
{
|
|
Console.WriteLine($"✓ {AreaCode} 도서관 선택 완료");
|
|
}
|
|
else
|
|
{
|
|
Console.WriteLine($"✗ {AreaCode} 도서관 선택 실패");
|
|
return false;
|
|
}
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
Console.WriteLine($"목표 도서관 선택 실패: {ex.Message}");
|
|
return false;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
Console.WriteLine("AreaCode가 비어있음 - 전체 도서관으로 검색");
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
Console.WriteLine($"도서관 선택 실패: {ex.Message}");
|
|
return false;
|
|
}
|
|
}
|
|
protected void SafeClick(IWebElement element)
|
|
{
|
|
// 안정적인 클릭을 위한 여러 방법 시도
|
|
try
|
|
{
|
|
// 1. 요소가 보이도록 스크롤 후 일반 클릭
|
|
((IJavaScriptExecutor)_driver).ExecuteScript("arguments[0].scrollIntoView(true);", element);
|
|
Thread.Sleep(300);
|
|
element.Click();
|
|
}
|
|
catch
|
|
{
|
|
try
|
|
{
|
|
// 2. JavaScript로 클릭 시도
|
|
((IJavaScriptExecutor)_driver).ExecuteScript("arguments[0].click();", element);
|
|
}
|
|
catch
|
|
{
|
|
try
|
|
{
|
|
// 3. Actions 클래스 사용
|
|
var actions = new Actions(_driver);
|
|
actions.MoveToElement(element).Click().Perform();
|
|
}
|
|
catch
|
|
{
|
|
try
|
|
{
|
|
// 4. 체크박스의 경우 직접 체크 상태 변경
|
|
if (element.TagName.ToLower() == "input" && element.GetAttribute("type") == "checkbox")
|
|
{
|
|
((IJavaScriptExecutor)_driver).ExecuteScript("arguments[0].checked = !arguments[0].checked;", element);
|
|
}
|
|
else
|
|
{
|
|
// 일반 요소는 강제 클릭
|
|
((IJavaScriptExecutor)_driver).ExecuteScript("arguments[0].dispatchEvent(new Event('click'));", element);
|
|
}
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
Console.WriteLine($"모든 클릭 방법 실패: {ex.Message}");
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
public async Task<BookSearchResult> SearchAsync(string searchTerm)
|
|
{
|
|
var result = new BookSearchResult
|
|
{
|
|
SiteName = SiteName,
|
|
SearchTerm = searchTerm,
|
|
SearchTime = DateTime.Now
|
|
};
|
|
|
|
try
|
|
{
|
|
// 드라이버가 없으면 자동으로 시작
|
|
if (_driver == null)
|
|
{
|
|
await StartDriver();
|
|
}
|
|
|
|
var cururl = _driver.Url;
|
|
if (cururl.Equals(SiteUrl) == false)
|
|
_driver.Navigate().GoToUrl(SiteUrl);
|
|
|
|
// 페이지 로딩 대기
|
|
var wait = new WebDriverWait(_driver, TimeSpan.FromSeconds(15));
|
|
|
|
// 페이지 로드 후 브라우저 배율을 80%로 설정 (브라우저가 표시되는 경우에만)
|
|
try
|
|
{
|
|
if (_driver.Manage().Window.Size.Width > 0) // 브라우저가 표시되는 경우
|
|
{
|
|
((IJavaScriptExecutor)_driver).ExecuteScript("document.body.style.zoom = '0.8';");
|
|
Console.WriteLine("페이지 로드 후 브라우저 배율을 80%로 설정했습니다.");
|
|
}
|
|
}
|
|
catch (Exception zoomEx)
|
|
{
|
|
Console.WriteLine($"페이지 배율 설정 실패: {zoomEx.Message}");
|
|
}
|
|
|
|
// 도서관 선택
|
|
if (SelectLibrary(wait) == false)
|
|
{
|
|
result.ErrorMessage = "도서관선택실패";
|
|
result.BookCount = -1;
|
|
result.IsSuccess = false;
|
|
return result;
|
|
}
|
|
|
|
// 검색어 입력
|
|
try
|
|
{
|
|
// 검색 입력창이 상호작용 가능할 때까지 대기
|
|
var searchInput = wait.Until(d => d.FindElement(By.Id("advTitle")));
|
|
var dispaly = searchInput.Displayed;
|
|
var enabled = searchInput.Enabled;
|
|
|
|
// 요소가 보이도록 스크롤
|
|
((IJavaScriptExecutor)_driver).ExecuteScript("arguments[0].scrollIntoView(true);", searchInput);
|
|
Thread.Sleep(300);
|
|
|
|
// readonly 속성이 있다면 제거
|
|
((IJavaScriptExecutor)_driver).ExecuteScript("arguments[0].removeAttribute('readonly');", searchInput);
|
|
|
|
// 클릭해서 포커스 맞추기
|
|
try
|
|
{
|
|
searchInput.Click();
|
|
Thread.Sleep(200);
|
|
}
|
|
catch { }
|
|
|
|
// 기존 값 제거 (여러 방법 시도)
|
|
try
|
|
{
|
|
searchInput.Clear();
|
|
}
|
|
catch
|
|
{
|
|
// Clear가 실패하면 JavaScript로 값 제거
|
|
((IJavaScriptExecutor)_driver).ExecuteScript("arguments[0].value = '';", searchInput);
|
|
}
|
|
|
|
// 검색어 입력 (여러 방법 시도)
|
|
try
|
|
{
|
|
searchInput.SendKeys(searchTerm);
|
|
}
|
|
catch
|
|
{
|
|
// SendKeys가 실패하면 JavaScript로 값 설정
|
|
((IJavaScriptExecutor)_driver).ExecuteScript("arguments[0].value = arguments[1];", searchInput, searchTerm);
|
|
|
|
// input 이벤트 발생시켜서 웹페이지에 변경사항 알림
|
|
((IJavaScriptExecutor)_driver).ExecuteScript("arguments[0].dispatchEvent(new Event('input'));", searchInput);
|
|
}
|
|
|
|
Console.WriteLine($"검색어 '{searchTerm}' 입력 완료");
|
|
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
result.ErrorMessage = $"검색어입력실패({ex.Message})";
|
|
result.BookCount = -1;
|
|
result.IsSuccess = false;
|
|
return result;
|
|
}
|
|
|
|
// 검색 버튼 클릭
|
|
try
|
|
{
|
|
Console.WriteLine("검색 버튼 찾기 시작...");
|
|
|
|
((IJavaScriptExecutor)_driver).ExecuteScript(@"
|
|
// 검색 버튼을 여러 방법으로 찾아서 클릭
|
|
var searchButtons = document.querySelectorAll('a.btTxt.bg.primary');
|
|
for (var i = 0; i < searchButtons.length; i++) {
|
|
var btn = searchButtons[i];
|
|
if (btn.innerText.trim() === '검색' || btn.innerText.includes('검색')) {
|
|
btn.click();
|
|
console.log('검색 버튼 클릭 성공');
|
|
break;
|
|
}
|
|
}
|
|
|
|
// 위 방법이 실패하면 마지막 버튼 클릭
|
|
if (searchButtons.length >= 2) {
|
|
searchButtons[searchButtons.length - 1].click();
|
|
console.log('마지막 버튼 클릭 성공');
|
|
}
|
|
");
|
|
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
result.ErrorMessage = $"검색버튼클릭실패({ex.Message})";
|
|
result.BookCount = -1;
|
|
result.IsSuccess = false;
|
|
return result;
|
|
}
|
|
|
|
SearchComplete:
|
|
Console.WriteLine("검색 실행 완료, 페이지 로딩 대기 중...");
|
|
|
|
|
|
if (this._driver.Url.EndsWith("DetailSearchResult") == false)
|
|
{
|
|
while (true)
|
|
{
|
|
await Task.Delay(100);
|
|
}
|
|
}
|
|
|
|
|
|
// 페이지 변경을 감지하는 메서드
|
|
await WaitForPageChange(new WebDriverWait(_driver, TimeSpan.FromSeconds(15)));
|
|
|
|
// 검색 결과 수 추출
|
|
var resultCount = ExtractBookCount(_driver, searchTerm, out string ermsg);
|
|
if (resultCount == -1)
|
|
{
|
|
result.BookCount = 0;
|
|
result.IsSuccess = false;
|
|
result.ErrorMessage = ermsg;
|
|
}
|
|
else
|
|
{
|
|
result.BookCount = resultCount;
|
|
result.IsSuccess = true;
|
|
result.ErrorMessage = ermsg;
|
|
}
|
|
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
result.IsSuccess = false;
|
|
result.ErrorMessage = ex.Message;
|
|
result.BookCount = 0;
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
private int ExtractBookCount(IWebDriver driver, string searchTerm, out string errmessage)
|
|
{
|
|
errmessage = string.Empty;
|
|
|
|
if (driver.Url.EndsWith("DetailSearchResult") == false)
|
|
{
|
|
errmessage = "결과페이지가아님";
|
|
return -1;
|
|
}
|
|
|
|
|
|
try
|
|
{
|
|
// 1. 검색결과가 없는 경우 확인
|
|
try
|
|
{
|
|
var noResultElements = driver.FindElements(By.XPath("//*[contains(text(), '검색결과가 없습니다') or contains(text(), '검색된 자료가 없습니다')]"));
|
|
if (noResultElements.Count > 0)
|
|
{
|
|
errmessage = "검색결과없음";
|
|
return 0;
|
|
}
|
|
}
|
|
catch
|
|
{
|
|
// 검색결과가 있는 경우로 진행
|
|
}
|
|
|
|
|
|
RetryPoint:
|
|
bool retry = false;
|
|
// 2. 안산시 도서관 특화: span.count.gothic em 요소에서 직접 추출
|
|
try
|
|
{
|
|
|
|
|
|
var countElement = driver.FindElement(By.CssSelector("span.count.gothic em"));
|
|
if (countElement != null)
|
|
{
|
|
// Text 대신 InnerHtml이나 GetAttribute 사용해보기
|
|
var countText = countElement.Text.Trim();
|
|
var innerHTML = countElement.GetAttribute("innerHTML");
|
|
var outerHTML = countElement.GetAttribute("outerHTML");
|
|
|
|
Console.WriteLine($"count 요소 .Text: '{countText}'");
|
|
Console.WriteLine($"count 요소 innerHTML: '{innerHTML}'");
|
|
Console.WriteLine($"count 요소 outerHTML: '{outerHTML}'");
|
|
|
|
// innerHTML이나 outerHTML에서 텍스트 추출 시도
|
|
string textToUse = !string.IsNullOrEmpty(countText) ? countText : innerHTML;
|
|
if (string.IsNullOrEmpty(textToUse))
|
|
{
|
|
textToUse = outerHTML;
|
|
}
|
|
|
|
Console.WriteLine($"추출할 텍스트: '{textToUse}'");
|
|
|
|
// "총 0 건 " 형태에서 숫자 추출
|
|
var match = Regex.Match(textToUse, @"총\s*(\d+)\s*건", RegexOptions.IgnoreCase);
|
|
if (match.Success && int.TryParse(match.Groups[1].Value, out int count))
|
|
{
|
|
if (count == 0)
|
|
{
|
|
errmessage = "검색결과없음";
|
|
Console.WriteLine("검색 결과: 0건");
|
|
return 0;
|
|
}
|
|
errmessage = $"검색성공({count}권)";
|
|
Console.WriteLine($"검색 결과: {count}건");
|
|
return count;
|
|
}
|
|
else
|
|
{
|
|
Console.WriteLine($"정규식 매칭 실패: '{textToUse}'");
|
|
}
|
|
}
|
|
}
|
|
catch (Exception ex1)
|
|
{
|
|
Console.WriteLine($"span.count.gothic em 요소 추출 실패: {ex1.Message}");
|
|
}
|
|
|
|
// 3. span.count 전체에서 추출 시도
|
|
try
|
|
{
|
|
var countSpan = driver.FindElement(By.CssSelector("span.count"));
|
|
if (countSpan != null)
|
|
{
|
|
var fullText = countSpan.Text.Trim();
|
|
Console.WriteLine($"count span 전체 텍스트: '{fullText}'");
|
|
|
|
var match = Regex.Match(fullText, @"총\s*(\d+)\s*건", RegexOptions.IgnoreCase);
|
|
if (match.Success && int.TryParse(match.Groups[1].Value, out int count))
|
|
{
|
|
if (count == 0)
|
|
{
|
|
errmessage = "검색결과없음";
|
|
return 0;
|
|
}
|
|
errmessage = $"검색성공({count}권)";
|
|
return count;
|
|
}
|
|
}
|
|
}
|
|
catch (Exception ex2)
|
|
{
|
|
Console.WriteLine($"span.count 요소 추출 실패: {ex2.Message}");
|
|
}
|
|
|
|
// 3. 페이지 소스에서 정규식으로 추출 시도
|
|
var pageSource = driver.PageSource;
|
|
|
|
// 검색 결과가 없다는 메시지 확인
|
|
if (pageSource.Contains("검색결과가 없습니다") ||
|
|
pageSource.Contains("검색된 자료가 없습니다") ||
|
|
pageSource.Contains("자료가 없습니다") ||
|
|
pageSource.Contains("총 0 건") ||
|
|
pageSource.Contains("총 0건"))
|
|
{
|
|
errmessage = "검색결과없음";
|
|
return 0;
|
|
}
|
|
|
|
// HTML에서 다양한 패턴 찾기 (안산시 도서관 특화)
|
|
var htmlPatterns = new[]
|
|
{
|
|
@"총\s*(\d+)\s*건",
|
|
@"<em[^>]*>총\s*(\d+)\s*건\s*</em>",
|
|
@"검색결과\s*:\s*총\s*(\d+)\s*건",
|
|
@"전체\s*(\d+)\s*건",
|
|
@"검색결과\s*총\s*(\d+)\s*건",
|
|
@"검색\s*결과\s*\(\s*총\s*(\d+)\s*건\s*\)",
|
|
@"총\s*<[^>]*>(\d+)</[^>]*>\s*건"
|
|
};
|
|
|
|
foreach (var pattern in htmlPatterns)
|
|
{
|
|
var match = Regex.Match(pageSource, pattern, RegexOptions.IgnoreCase);
|
|
if (match.Success)
|
|
{
|
|
if (int.TryParse(match.Groups[1].Value, out int count))
|
|
{
|
|
if (count == 0)
|
|
{
|
|
errmessage = "검색결과없음";
|
|
return 0;
|
|
}
|
|
errmessage = $"검색성공({count}권)";
|
|
return count;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (retry == false)
|
|
{
|
|
Console.WriteLine( "결과를 찾을 수 없어 재시도");
|
|
retry = true;
|
|
Task.Delay(1000);
|
|
goto RetryPoint;
|
|
}
|
|
else
|
|
{
|
|
errmessage = "결과수량을찾을수없음";
|
|
return -1;
|
|
}
|
|
|
|
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
errmessage = ex.Message;
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
// 페이지 변경을 감지하는 메서드
|
|
public async Task WaitForPageChange(WebDriverWait wait)
|
|
{
|
|
try
|
|
{
|
|
await Task.Delay(500);
|
|
|
|
|
|
|
|
// 페이지 로딩 상태 확인
|
|
wait.Until(d =>
|
|
{
|
|
var readyState = ((IJavaScriptExecutor)d).ExecuteScript("return document.readyState");
|
|
return readyState.Equals("complete");
|
|
});
|
|
|
|
// 검색 결과 페이지가 로드될 때까지 대기
|
|
wait.Until(d =>
|
|
{
|
|
try
|
|
{
|
|
var pageSource = d.PageSource;
|
|
// 검색 결과나 관련 요소가 나타나면 로드 완료
|
|
return pageSource.Contains("검색결과") ||
|
|
pageSource.Contains("총") ||
|
|
pageSource.Contains("검색결과가 없습니다") ||
|
|
pageSource.Contains("건");
|
|
}
|
|
catch
|
|
{
|
|
return false;
|
|
}
|
|
});
|
|
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
// 모든 감지 방법이 실패하면 최소한의 대기 시간 적용
|
|
await Task.Delay(3000);
|
|
Console.WriteLine($"페이지 변경 감지 실패: {ex.Message}");
|
|
}
|
|
}
|
|
}
|
|
} |