From b364ffc054e220aa1a6ae49d09fb90183bd99fdb Mon Sep 17 00:00:00 2001 From: "Arin(asus)" Date: Thu, 14 Aug 2025 16:23:46 +0900 Subject: [PATCH] =?UTF-8?q?feat:=20=ED=81=AC=EB=A1=A4=EB=A7=81=201?= =?UTF-8?q?=EC=B0=A8=20=EC=99=84=EB=A3=8C=20-=20400=EA=B0=9C=20=EC=9D=B4?= =?UTF-8?q?=EC=83=81=20=EB=8F=84=EC=84=9C=EA=B4=80=20=EB=AA=A9=EB=A1=9D=20?= =?UTF-8?q?=EC=99=84=EC=84=B1?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit 추가된 도서관 시스템: • 광주남구도서관 (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 --- .../unimarc/SearchModel/AnsanLibSearcher.cs | 2 +- .../unimarc/SearchModel/BukguLibSearcher.cs | 2 +- .../SearchModel/ChosunTechLibSearcher.cs | 201 +++++++++ .../SearchModel/ChosunUnivLibSearcher.cs | 201 +++++++++ unimarc/unimarc/SearchModel/DLSSearcher.cs | 1 + .../unimarc/SearchModel/GoheungLibSearcher.cs | 2 +- .../SearchModel/GwangjuCityLibSearcher.cs | 2 +- .../SearchModel/GwangjuDongguLibSearcher.cs | 2 +- .../SearchModel/GwangjuSeoguLibSearcher.cs | 2 +- .../SearchModel/GwangsanLibSearcher.cs | 2 +- .../SearchModel/GyeongnamLibSearcher.cs | 2 +- .../unimarc/SearchModel/ILibrarySearcher.cs | 1 + .../unimarc/SearchModel/IksanLibSearcher.cs | 2 +- .../SearchModel/JeonbukEduLibSearcher.cs | 2 +- .../SearchModel/JunnamEduJiheaNuriSearcher.cs | 2 +- .../unimarc/SearchModel/JunnamEduSearcher.cs | 2 +- unimarc/unimarc/SearchModel/KcmLibSearcher.cs | 217 +++++++++ .../KwangjuCityEduLibrarySearcher.cs | 2 +- .../unimarc/SearchModel/MokpoLibSearcher.cs | 2 +- .../unimarc/SearchModel/MuanLibSearcher.cs | 416 ++++++++++++++++++ .../SearchModel/NamguLibrarySearcher.cs | 2 +- .../SearchModel/SuncheonLibSearcher.cs | 2 +- .../unimarc/SearchModel/WandoLibSearcher.cs | 2 +- .../unimarc/SearchModel/YeosuLibSearcher.cs | 2 +- unimarc/unimarc/UniMarc.csproj | 4 + unimarc/unimarc/마크/Check_copyWD.cs | 51 ++- 26 files changed, 1109 insertions(+), 19 deletions(-) create mode 100644 unimarc/unimarc/SearchModel/ChosunTechLibSearcher.cs create mode 100644 unimarc/unimarc/SearchModel/ChosunUnivLibSearcher.cs create mode 100644 unimarc/unimarc/SearchModel/KcmLibSearcher.cs create mode 100644 unimarc/unimarc/SearchModel/MuanLibSearcher.cs diff --git a/unimarc/unimarc/SearchModel/AnsanLibSearcher.cs b/unimarc/unimarc/SearchModel/AnsanLibSearcher.cs index 7e8d0ce..395368d 100644 --- a/unimarc/unimarc/SearchModel/AnsanLibSearcher.cs +++ b/unimarc/unimarc/SearchModel/AnsanLibSearcher.cs @@ -25,7 +25,7 @@ namespace BokBonCheck { public class AnsanLibSearcher : ILibrarySearcher { - protected string AreaCode { get; set; } = string.Empty; + 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; diff --git a/unimarc/unimarc/SearchModel/BukguLibSearcher.cs b/unimarc/unimarc/SearchModel/BukguLibSearcher.cs index 9822a3c..76d7430 100644 --- a/unimarc/unimarc/SearchModel/BukguLibSearcher.cs +++ b/unimarc/unimarc/SearchModel/BukguLibSearcher.cs @@ -20,7 +20,7 @@ namespace BokBonCheck { public class BukguLibSearcher : ILibrarySearcher { - protected string AreaCode { get; set; } = string.Empty; + public string AreaCode { get; set; } = string.Empty; public string SiteName { get; protected set; } public string SiteUrl { get; protected set; } = "https://lib.bukgu.gwangju.kr/main/bookSearchSmartlib.do?PID=0301"; public bool HttpApiMode { get; set; } = false; diff --git a/unimarc/unimarc/SearchModel/ChosunTechLibSearcher.cs b/unimarc/unimarc/SearchModel/ChosunTechLibSearcher.cs new file mode 100644 index 0000000..8d1c14f --- /dev/null +++ b/unimarc/unimarc/SearchModel/ChosunTechLibSearcher.cs @@ -0,0 +1,201 @@ +using System; +using System.Net.Http; +using System.Threading.Tasks; +using System.Text.RegularExpressions; +using System.Web; +using UniMarc.SearchModel; +using System.Text; +using OpenQA.Selenium.Support.UI; + +namespace BokBonCheck +{ + public class ChosunTechLibSearcher : ILibrarySearcher + { + public string AreaCode { get; set; } = string.Empty; + public string SiteName { get; protected set; } + public string SiteUrl => "https://book.cst.ac.kr/Search/"; + public bool HttpApiMode { get; set; } = true; + + public int No { get; set; } + + private static readonly HttpClient _httpClient = new HttpClient() + { + DefaultRequestHeaders = + { + { "User-Agent", "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/120.0.0.0 Safari/537.36" } + } + }; + + public ChosunTechLibSearcher(int no, string areaCode, string areaName) + { + this.No = no; + this.AreaCode = areaCode; + this.SiteName = $"조선이공대학교({areaName})"; + } + + public async Task StartDriver(bool showdriver = false) + { + // HTTP 클라이언트 사용으로 별도 드라이버 불필요 + await Task.CompletedTask; + } + + public void StopDriver() + { + // HTTP 클라이언트 사용으로 별도 정리 불필요 + } + + public async Task SearchAsync(string searchTerm) + { + var result = new BookSearchResult + { + SiteName = SiteName, + SearchTerm = searchTerm, + SearchTime = DateTime.Now + }; + + try + { + // 검색어 URL 인코딩 + var encodedSearchTerm = HttpUtility.UrlEncode(searchTerm, Encoding.UTF8); + + // 검색 URL 구성 - GET 방식 (도서명 검색: st=tt) + var searchUrl = $"{SiteUrl}?q=tt:{encodedSearchTerm}&st=tt&searchTruncate=true&campuscode=00"; + + Console.WriteLine($"조선이공대학교도서관 검색 URL: {searchUrl}"); + + // HTTP GET 요청 실행 (추가 헤더 포함) + using (var request = new HttpRequestMessage(HttpMethod.Get, searchUrl)) + { + // 브라우저와 유사한 헤더 추가 + request.Headers.Add("Accept", "text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,*/*;q=0.8"); + request.Headers.Add("Accept-Language", "ko-KR,ko;q=0.8,en-US;q=0.5,en;q=0.3"); + request.Headers.Add("Accept-Encoding", "gzip, deflate, br"); + request.Headers.Add("Connection", "keep-alive"); + request.Headers.Add("Upgrade-Insecure-Requests", "1"); + + var response = await _httpClient.SendAsync(request); + + if (!response.IsSuccessStatusCode) + { + var errorContent = await response.Content.ReadAsStringAsync(); + throw new HttpRequestException($"HTTP {(int)response.StatusCode} {response.StatusCode}: {errorContent}"); + } + + var htmlContent = await response.Content.ReadAsStringAsync(); + + // 검색 결과 수 추출 + var resultCount = ExtractBookCount(htmlContent, out string errorMessage); + + if (resultCount == -1) + { + result.BookCount = 0; + result.IsSuccess = false; + result.ErrorMessage = errorMessage; + } + else + { + result.BookCount = resultCount; + result.IsSuccess = true; + result.ErrorMessage = $"검색성공({resultCount}권)"; + } + } + + } + catch (Exception ex) + { + result.IsSuccess = false; + result.ErrorMessage = $"검색 오류: {ex.Message}"; + result.BookCount = 0; + Console.WriteLine($"조선이공대학교도서관 검색 오류: {ex.Message}"); + } + + return result; + } + + private int ExtractBookCount(string htmlContent, out string errorMessage) + { + errorMessage = string.Empty; + + try + { + // 1. 검색 결과가 없는 경우 확인 + if (htmlContent.Contains("검색결과가 없습니다") || + htmlContent.Contains("검색된 자료가 없습니다") || + htmlContent.Contains("자료가 없습니다") || + htmlContent.Contains("0 Results:")) + { + errorMessage = "검색결과없음"; + return 0; + } + + // 2. sponge-search-top-header-left에서 결과 수량 추출: 27 Results: + var patterns = new[] + { + @"]*class=""pull-left\s+sponge-search-top-header-left""[^>]*>.*?\s*(\d+)\s*\s*Results:", + @"sponge-search-top-header-left[^>]*>.*?\s*(\d+)\s*\s*Results:", + @"\s*(\d+)\s*\s*Results:", + @"pull-left\s+sponge-search-top-header-left[^>]*>.*?\s*(\d+)\s*" + }; + + foreach (var pattern in patterns) + { + var match = Regex.Match(htmlContent, pattern, RegexOptions.IgnoreCase | RegexOptions.Singleline); + if (match.Success) + { + if (int.TryParse(match.Groups[1].Value, out int count)) + { + if (count == 0) + { + errorMessage = "검색결과없음"; + return 0; + } + errorMessage = $"검색성공({count}권)"; + Console.WriteLine($"조선이공대학교도서관 검색 결과: {count}건"); + return count; + } + } + } + + // 3. 더 자세한 패턴으로 시도 (줄바꿈 포함) + var multilinePatterns = new[] + { + @"]*class=""pull-left\s+sponge-search-top-header-left""[^>]*>\s*\r?\n?\s*\s*\r?\n?\s*(\d+)\s*\r?\n?\s*\s*Results:", + @"\s*\r?\n?\s*(\d+)\s*\r?\n?\s*\s*Results:" + }; + + foreach (var pattern in multilinePatterns) + { + var match = Regex.Match(htmlContent, pattern, RegexOptions.IgnoreCase | RegexOptions.Multiline | RegexOptions.Singleline); + if (match.Success) + { + if (int.TryParse(match.Groups[1].Value, out int count)) + { + if (count == 0) + { + errorMessage = "검색결과없음"; + return 0; + } + errorMessage = $"검색성공({count}권)"; + Console.WriteLine($"조선이공대학교도서관 검색 결과: {count}건"); + return count; + } + } + } + + errorMessage = "검색결과 패턴을 찾을 수 없음"; + Console.WriteLine("조선이공대학교도서관 검색결과 패턴을 찾을 수 없음"); + return -1; + } + catch (Exception ex) + { + errorMessage = $"결과 분석 오류: {ex.Message}"; + return -1; + } + } + + public Task WaitForPageChange(WebDriverWait wait) + { + throw new NotImplementedException(); + } + } +} \ No newline at end of file diff --git a/unimarc/unimarc/SearchModel/ChosunUnivLibSearcher.cs b/unimarc/unimarc/SearchModel/ChosunUnivLibSearcher.cs new file mode 100644 index 0000000..d9c7536 --- /dev/null +++ b/unimarc/unimarc/SearchModel/ChosunUnivLibSearcher.cs @@ -0,0 +1,201 @@ +using System; +using System.Net.Http; +using System.Threading.Tasks; +using System.Text.RegularExpressions; +using System.Web; +using UniMarc.SearchModel; +using System.Text; +using OpenQA.Selenium.Support.UI; + +namespace BokBonCheck +{ + public class ChosunUnivLibSearcher : ILibrarySearcher + { + public string AreaCode { get; set; } = string.Empty; + public string SiteName { get; protected set; } + public string SiteUrl => "https://library.chosun.ac.kr/search/laz/result"; + public bool HttpApiMode { get; set; } = true; + + public int No { get; set; } + + private static readonly HttpClient _httpClient = new HttpClient() + { + DefaultRequestHeaders = + { + { "User-Agent", "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/120.0.0.0 Safari/537.36" } + } + }; + + public ChosunUnivLibSearcher(int no, string areaCode, string areaName) + { + this.No = no; + this.AreaCode = areaCode; + this.SiteName = $"조선대학교중앙({areaName})"; + } + + public async Task StartDriver(bool showdriver = false) + { + // HTTP 클라이언트 사용으로 별도 드라이버 불필요 + await Task.CompletedTask; + } + + public void StopDriver() + { + // HTTP 클라이언트 사용으로 별도 정리 불필요 + } + + public async Task SearchAsync(string searchTerm) + { + var result = new BookSearchResult + { + SiteName = SiteName, + SearchTerm = searchTerm, + SearchTime = DateTime.Now + }; + + try + { + // 검색어 URL 인코딩 + var encodedSearchTerm = HttpUtility.UrlEncode(searchTerm, Encoding.UTF8); + + // 검색 URL 구성 - GET 방식 (서명전용 검색: si=1) + var searchUrl = $"{SiteUrl}?st=KWRD&si=1&q={encodedSearchTerm}&folder_id=null"; + + Console.WriteLine($"조선대학교중앙도서관 검색 URL: {searchUrl}"); + + // HTTP GET 요청 실행 (추가 헤더 포함) + using (var request = new HttpRequestMessage(HttpMethod.Get, searchUrl)) + { + // 브라우저와 유사한 헤더 추가 + request.Headers.Add("Accept", "text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,*/*;q=0.8"); + request.Headers.Add("Accept-Language", "ko-KR,ko;q=0.8,en-US;q=0.5,en;q=0.3"); + request.Headers.Add("Accept-Encoding", "gzip, deflate, br"); + request.Headers.Add("Connection", "keep-alive"); + request.Headers.Add("Upgrade-Insecure-Requests", "1"); + + var response = await _httpClient.SendAsync(request); + + if (!response.IsSuccessStatusCode) + { + var errorContent = await response.Content.ReadAsStringAsync(); + throw new HttpRequestException($"HTTP {(int)response.StatusCode} {response.StatusCode}: {errorContent}"); + } + + var htmlContent = await response.Content.ReadAsStringAsync(); + + // 검색 결과 수 추출 + var resultCount = ExtractBookCount(htmlContent, out string errorMessage); + + if (resultCount == -1) + { + result.BookCount = 0; + result.IsSuccess = false; + result.ErrorMessage = errorMessage; + } + else + { + result.BookCount = resultCount; + result.IsSuccess = true; + result.ErrorMessage = $"검색성공({resultCount}권)"; + } + } + + } + catch (Exception ex) + { + result.IsSuccess = false; + result.ErrorMessage = $"검색 오류: {ex.Message}"; + result.BookCount = 0; + Console.WriteLine($"조선대학교중앙도서관 검색 오류: {ex.Message}"); + } + + return result; + } + + private int ExtractBookCount(string htmlContent, out string errorMessage) + { + errorMessage = string.Empty; + + try + { + // 1. 검색 결과가 없는 경우 확인 + if (htmlContent.Contains("검색결과가 없습니다") || + htmlContent.Contains("검색된 자료가 없습니다") || + htmlContent.Contains("자료가 없습니다") || + htmlContent.Contains("총 0건")) + { + errorMessage = "검색결과없음"; + return 0; + } + + // 2. searchCnt에서 결과 수량 추출: 총 7건 중 7건 출력 + var patterns = new[] + { + @"]*class=""searchCnt""[^>]*>.*?총\s*\s*(\d+)\s*\s*건", + @"총\s*\s*(\d+)\s*\s*건\s*중", + @"총\s*\s*(\d+)\s*\s*건", + @"\s*(\d+)\s*\s*건\s*중\s*\s*(\d+)\s*\s*건\s*출력" + }; + + foreach (var pattern in patterns) + { + var match = Regex.Match(htmlContent, pattern, RegexOptions.IgnoreCase); + if (match.Success) + { + if (int.TryParse(match.Groups[1].Value, out int count)) + { + if (count == 0) + { + errorMessage = "검색결과없음"; + return 0; + } + errorMessage = $"검색성공({count}권)"; + Console.WriteLine($"조선대학교중앙도서관 검색 결과: {count}건"); + return count; + } + } + } + + // 3. 더 자세한 패턴으로 시도 (줄바꿈 포함) + var multilinePatterns = new[] + { + @"]*class=""searchCnt""[^>]*>\s*\r?\n?\s*총\s*\s*\r?\n?\s*(\d+)\s*\r?\n?\s*\s*건", + @"총\s*\s*\r?\n?\s*(\d+)\s*\r?\n?\s*\s*건" + }; + + foreach (var pattern in multilinePatterns) + { + var match = Regex.Match(htmlContent, pattern, RegexOptions.IgnoreCase | RegexOptions.Multiline); + if (match.Success) + { + if (int.TryParse(match.Groups[1].Value, out int count)) + { + if (count == 0) + { + errorMessage = "검색결과없음"; + return 0; + } + errorMessage = $"검색성공({count}권)"; + Console.WriteLine($"조선대학교중앙도서관 검색 결과: {count}건"); + return count; + } + } + } + + errorMessage = "검색결과 패턴을 찾을 수 없음"; + Console.WriteLine("조선대학교중앙도서관 검색결과 패턴을 찾을 수 없음"); + return -1; + } + catch (Exception ex) + { + errorMessage = $"결과 분석 오류: {ex.Message}"; + return -1; + } + } + + public Task WaitForPageChange(WebDriverWait wait) + { + throw new NotImplementedException(); + } + } +} \ No newline at end of file diff --git a/unimarc/unimarc/SearchModel/DLSSearcher.cs b/unimarc/unimarc/SearchModel/DLSSearcher.cs index 5681e02..9bd23ae 100644 --- a/unimarc/unimarc/SearchModel/DLSSearcher.cs +++ b/unimarc/unimarc/SearchModel/DLSSearcher.cs @@ -18,6 +18,7 @@ namespace BokBonCheck { public class DLSSearcher : ILibrarySearcher { + public string AreaCode { get; set; } = string.Empty; public bool HttpApiMode { get; set; } = false; public string SiteName { get; protected set; } = "DLS"; public string SiteUrl => "https://dls1.edunet.net/DLS/bookMng/bookMain"; diff --git a/unimarc/unimarc/SearchModel/GoheungLibSearcher.cs b/unimarc/unimarc/SearchModel/GoheungLibSearcher.cs index 92ec876..fde295f 100644 --- a/unimarc/unimarc/SearchModel/GoheungLibSearcher.cs +++ b/unimarc/unimarc/SearchModel/GoheungLibSearcher.cs @@ -19,7 +19,7 @@ namespace BokBonCheck { public class GoheungLibSearcher : ILibrarySearcher { - protected string AreaCode { get; set; } = string.Empty; + public string AreaCode { get; set; } = string.Empty; public string SiteName { get; protected set; } public string SiteUrl => "https://www.ghlib.go.kr/BookSearch/detail"; public bool HttpApiMode { get; set; } = false; diff --git a/unimarc/unimarc/SearchModel/GwangjuCityLibSearcher.cs b/unimarc/unimarc/SearchModel/GwangjuCityLibSearcher.cs index 9c702dd..b419b4e 100644 --- a/unimarc/unimarc/SearchModel/GwangjuCityLibSearcher.cs +++ b/unimarc/unimarc/SearchModel/GwangjuCityLibSearcher.cs @@ -19,7 +19,7 @@ namespace BokBonCheck { public class GwangjuCityLibSearcher : ILibrarySearcher { - protected string AreaCode { get; set; } = string.Empty; + public string AreaCode { get; set; } = string.Empty; public string SiteName { get; protected set; } public string SiteUrl => "https://citylib.gwangju.go.kr/main/bookSearch"; public bool HttpApiMode { get; set; } = false; diff --git a/unimarc/unimarc/SearchModel/GwangjuDongguLibSearcher.cs b/unimarc/unimarc/SearchModel/GwangjuDongguLibSearcher.cs index 96064fe..5f16854 100644 --- a/unimarc/unimarc/SearchModel/GwangjuDongguLibSearcher.cs +++ b/unimarc/unimarc/SearchModel/GwangjuDongguLibSearcher.cs @@ -11,7 +11,7 @@ namespace BokBonCheck { public class GwangjuDongguLibSearcher : ILibrarySearcher { - protected string AreaCode { get; set; } = string.Empty; + public string AreaCode { get; set; } = string.Empty; public string SiteName { get; protected set; } public string SiteUrl => "https://lib.donggu.kr/BookSearch/detail"; public bool HttpApiMode { get; set; } = true; diff --git a/unimarc/unimarc/SearchModel/GwangjuSeoguLibSearcher.cs b/unimarc/unimarc/SearchModel/GwangjuSeoguLibSearcher.cs index 167f09c..c0ea2bc 100644 --- a/unimarc/unimarc/SearchModel/GwangjuSeoguLibSearcher.cs +++ b/unimarc/unimarc/SearchModel/GwangjuSeoguLibSearcher.cs @@ -23,7 +23,7 @@ namespace BokBonCheck { public class GwangjuSeoguLibSearcher : ILibrarySearcher { - protected string AreaCode { get; set; } = string.Empty; + public string AreaCode { get; set; } = string.Empty; public string SiteName { get; protected set; } public virtual string SiteUrl => "https://library.seogu.gwangju.kr/index.9is?contentUid=9be5df897834aa07017868116d3407de"; public bool HttpApiMode { get; set; } = false; diff --git a/unimarc/unimarc/SearchModel/GwangsanLibSearcher.cs b/unimarc/unimarc/SearchModel/GwangsanLibSearcher.cs index 8cf00e1..d32cd53 100644 --- a/unimarc/unimarc/SearchModel/GwangsanLibSearcher.cs +++ b/unimarc/unimarc/SearchModel/GwangsanLibSearcher.cs @@ -10,7 +10,7 @@ namespace BokBonCheck { public class GwangsanLibSearcher : ILibrarySearcher { - protected string AreaCode { get; set; } = string.Empty; + public string AreaCode { get; set; } = string.Empty; public string SiteName { get; protected set; } public string SiteUrl => "https://lib.gwangsan.go.kr/main/bookSearch/advanced"; public bool HttpApiMode { get; set; } = true; diff --git a/unimarc/unimarc/SearchModel/GyeongnamLibSearcher.cs b/unimarc/unimarc/SearchModel/GyeongnamLibSearcher.cs index 6ca0f05..1cce63c 100644 --- a/unimarc/unimarc/SearchModel/GyeongnamLibSearcher.cs +++ b/unimarc/unimarc/SearchModel/GyeongnamLibSearcher.cs @@ -14,7 +14,7 @@ namespace BokBonCheck { public class GyeongnamLibSearcher : ILibrarySearcher { - protected string AreaCode { get; set; } = string.Empty; + public string AreaCode { get; set; } = string.Empty; public string SiteName { get; protected set; } public string SiteUrl => "https://lib.gyeongnam.go.kr/index.lib?menuCd=DOM_000000201012000000"; public bool HttpApiMode { get; set; } = false; diff --git a/unimarc/unimarc/SearchModel/ILibrarySearcher.cs b/unimarc/unimarc/SearchModel/ILibrarySearcher.cs index ee33108..addcae2 100644 --- a/unimarc/unimarc/SearchModel/ILibrarySearcher.cs +++ b/unimarc/unimarc/SearchModel/ILibrarySearcher.cs @@ -15,6 +15,7 @@ namespace BokBonCheck /// ũѸ ƴ HTTP ȣ ó˴ϴ /// bool HttpApiMode { get; set; } + string AreaCode { get; set; } int No { get; set; } string SiteName { get; } string SiteUrl { get; } diff --git a/unimarc/unimarc/SearchModel/IksanLibSearcher.cs b/unimarc/unimarc/SearchModel/IksanLibSearcher.cs index e5f83e9..69750ee 100644 --- a/unimarc/unimarc/SearchModel/IksanLibSearcher.cs +++ b/unimarc/unimarc/SearchModel/IksanLibSearcher.cs @@ -9,7 +9,7 @@ namespace BokBonCheck { public class IksanLibSearcher : ILibrarySearcher { - protected string AreaCode { get; set; } = string.Empty; + public string AreaCode { get; set; } = string.Empty; public string SiteName { get; protected set; } public string SiteUrl => "https://lib.iksan.go.kr/main/site/search/bookSearch.do"; public bool HttpApiMode { get; set; } = true; diff --git a/unimarc/unimarc/SearchModel/JeonbukEduLibSearcher.cs b/unimarc/unimarc/SearchModel/JeonbukEduLibSearcher.cs index e5c53f7..71f5208 100644 --- a/unimarc/unimarc/SearchModel/JeonbukEduLibSearcher.cs +++ b/unimarc/unimarc/SearchModel/JeonbukEduLibSearcher.cs @@ -19,7 +19,7 @@ namespace BokBonCheck { public class JeonbukEduLibSearcher : ILibrarySearcher { - protected string AreaCode { get; set; } = string.Empty; + public string AreaCode { get; set; } = string.Empty; public string SiteName { get; protected set; } public string SiteUrl => "https://lib.jbe.go.kr/jbe/intro/search/index.do"; public bool HttpApiMode { get; set; } = false; diff --git a/unimarc/unimarc/SearchModel/JunnamEduJiheaNuriSearcher.cs b/unimarc/unimarc/SearchModel/JunnamEduJiheaNuriSearcher.cs index 91bf525..22f3904 100644 --- a/unimarc/unimarc/SearchModel/JunnamEduJiheaNuriSearcher.cs +++ b/unimarc/unimarc/SearchModel/JunnamEduJiheaNuriSearcher.cs @@ -20,7 +20,7 @@ namespace BokBonCheck public class JunnamEduJiheaNuriSearcher : ILibrarySearcher { - protected string AreaCode { get; set; } = string.Empty; + public string AreaCode { get; set; } = string.Empty; public string SiteName { get; protected set; } public string SiteUrl => "https://jnelib.jne.go.kr/book/search_book/search.es?mid=d20101000000"; public bool HttpApiMode { get; set; } = false; diff --git a/unimarc/unimarc/SearchModel/JunnamEduSearcher.cs b/unimarc/unimarc/SearchModel/JunnamEduSearcher.cs index b8f34d5..e440bbc 100644 --- a/unimarc/unimarc/SearchModel/JunnamEduSearcher.cs +++ b/unimarc/unimarc/SearchModel/JunnamEduSearcher.cs @@ -19,7 +19,7 @@ namespace BokBonCheck public class JunnamEduSearcher : ILibrarySearcher { - protected string AreaCode { get; set; } = string.Empty; + public string AreaCode { get; set; } = string.Empty; public string SiteName { get; protected set; } public string SiteUrl => "https://jnelib.jne.go.kr/book/search_book/search.es?mid=d50101000000"; public bool HttpApiMode { get; set; } = false; diff --git a/unimarc/unimarc/SearchModel/KcmLibSearcher.cs b/unimarc/unimarc/SearchModel/KcmLibSearcher.cs new file mode 100644 index 0000000..654c050 --- /dev/null +++ b/unimarc/unimarc/SearchModel/KcmLibSearcher.cs @@ -0,0 +1,217 @@ +using System; +using System.Net.Http; +using System.Threading.Tasks; +using System.Text.RegularExpressions; +using System.Web; +using UniMarc.SearchModel; +using System.Text; +using OpenQA.Selenium.Support.UI; + +namespace BokBonCheck +{ + public class KcmLibSearcher : ILibrarySearcher + { + public string AreaCode { get; set; } = string.Empty; + public string SiteName { get; protected set; } + public string SiteUrl => "http://218.157.123.11:9996/kcms/KBookSearch/BookNomalSearch"; + public bool HttpApiMode { get; set; } = true; + + public int No { get; set; } + + private static readonly HttpClient _httpClient = new HttpClient() + { + DefaultRequestHeaders = + { + { "User-Agent", "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/120.0.0.0 Safari/537.36" } + } + }; + + public KcmLibSearcher(int no, string areaCode, string areaName) + { + this.No = no; + this.AreaCode = areaCode; + this.SiteName = $"KCMS({areaName})"; + } + + public async Task StartDriver(bool showdriver = false) + { + // HTTP 클라이언트 사용으로 별도 드라이버 불필요 + await Task.CompletedTask; + } + + public void StopDriver() + { + // HTTP 클라이언트 사용으로 별도 정리 불필요 + } + + public async Task SearchAsync(string searchTerm) + { + var result = new BookSearchResult + { + SiteName = SiteName, + SearchTerm = searchTerm, + SearchTime = DateTime.Now + }; + + try + { + // 검색어 URL 인코딩 + var encodedSearchTerm = HttpUtility.UrlEncode(searchTerm, Encoding.UTF8); + + // 도서관 코드가 없으면 기본값 사용 + var libCode = string.IsNullOrEmpty(AreaCode) ? "MA" : AreaCode; + + // 검색 URL 구성 - GET 방식 + var searchUrl = $"{SiteUrl}/{libCode}?" + + $"book_type=BOOK&" + + $"main_search_txt={encodedSearchTerm}&" + + $"search_txt={encodedSearchTerm}&" + + $"facetCol=&facetFlag=&" + + $"manage_code={libCode}&" + + $"pageno=1&display=10&" + + $"lib_manage_code=&all_check_flag=&book_type=&" + + $"detail_search_type=Nomal&" + + $"url_lib_code={libCode}&" + + $"input_search_text={encodedSearchTerm}&" + + $"manage_code=&option=&libcode=&order_by=&orderby=&" + + $"orderby_item=ACCURACY_SORT&" + + $"real_search_text={encodedSearchTerm}"; + + Console.WriteLine($"KCM자료검색시스템 검색 URL: {searchUrl}"); + + // HTTP GET 요청 실행 (추가 헤더 포함) + using (var request = new HttpRequestMessage(HttpMethod.Get, searchUrl)) + { + // 브라우저와 유사한 헤더 추가 + request.Headers.Add("Accept", "text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,*/*;q=0.8"); + request.Headers.Add("Accept-Language", "ko-KR,ko;q=0.8,en-US;q=0.5,en;q=0.3"); + request.Headers.Add("Accept-Encoding", "gzip, deflate, br"); + request.Headers.Add("Connection", "keep-alive"); + request.Headers.Add("Upgrade-Insecure-Requests", "1"); + + var response = await _httpClient.SendAsync(request); + + if (!response.IsSuccessStatusCode) + { + var errorContent = await response.Content.ReadAsStringAsync(); + throw new HttpRequestException($"HTTP {(int)response.StatusCode} {response.StatusCode}: {errorContent}"); + } + + var htmlContent = await response.Content.ReadAsStringAsync(); + + // 검색 결과 수 추출 + var resultCount = ExtractBookCount(htmlContent, out string errorMessage); + + if (resultCount == -1) + { + result.BookCount = 0; + result.IsSuccess = false; + result.ErrorMessage = errorMessage; + } + else + { + result.BookCount = resultCount; + result.IsSuccess = true; + result.ErrorMessage = $"검색성공({resultCount}권)"; + } + } + + } + catch (Exception ex) + { + result.IsSuccess = false; + result.ErrorMessage = $"검색 오류: {ex.Message}"; + result.BookCount = 0; + Console.WriteLine($"KCM자료검색시스템 검색 오류: {ex.Message}"); + } + + return result; + } + + private int ExtractBookCount(string htmlContent, out string errorMessage) + { + errorMessage = string.Empty; + + try + { + // 1. 검색 결과가 없는 경우 확인 + if (htmlContent.Contains("검색결과가 없습니다") || + htmlContent.Contains("검색된 자료가 없습니다") || + htmlContent.Contains("자료가 없습니다") || + htmlContent.Contains("총 0 건이 검색되었습니다")) + { + errorMessage = "검색결과없음"; + return 0; + } + + // 2. 검색결과에서 결과 수량 추출: 총 28 건이 검색되었습니다. + var patterns = new[] + { + @"

\s*검색결과.*?\s*총\s*(\d+)\s*건이\s*검색되었습니다\.\s*", + @"\s*총\s*(\d+)\s*건이\s*검색되었습니다\.\s*", + @"총\s*(\d+)\s*건이\s*검색되었습니다", + @"총\s*(\d+)\s*건" + }; + + foreach (var pattern in patterns) + { + var match = Regex.Match(htmlContent, pattern, RegexOptions.IgnoreCase | RegexOptions.Singleline); + if (match.Success) + { + if (int.TryParse(match.Groups[1].Value, out int count)) + { + if (count == 0) + { + errorMessage = "검색결과없음"; + return 0; + } + errorMessage = $"검색성공({count}권)"; + Console.WriteLine($"KCM자료검색시스템 검색 결과: {count}건"); + return count; + } + } + } + + // 3. 더 자세한 패턴으로 시도 (줄바꿈 포함) + var multilinePatterns = new[] + { + @"

\s*\r?\n?\s*검색결과.*?\s*\r?\n?\s*총\s*\r?\n?\s*(\d+)\s*\r?\n?\s*건이\s*검색되었습니다", + @"\s*\r?\n?\s*총\s*\r?\n?\s*(\d+)\s*\r?\n?\s*건이\s*검색되었습니다" + }; + + foreach (var pattern in multilinePatterns) + { + var match = Regex.Match(htmlContent, pattern, RegexOptions.IgnoreCase | RegexOptions.Multiline | RegexOptions.Singleline); + if (match.Success) + { + if (int.TryParse(match.Groups[1].Value, out int count)) + { + if (count == 0) + { + errorMessage = "검색결과없음"; + return 0; + } + errorMessage = $"검색성공({count}권)"; + Console.WriteLine($"KCM자료검색시스템 검색 결과: {count}건"); + return count; + } + } + } + + errorMessage = "검색결과 패턴을 찾을 수 없음"; + Console.WriteLine("KCM자료검색시스템 검색결과 패턴을 찾을 수 없음"); + return -1; + } + catch (Exception ex) + { + errorMessage = $"결과 분석 오류: {ex.Message}"; + return -1; + } + } + + public Task WaitForPageChange(WebDriverWait wait) + { + throw new NotImplementedException(); + } + } +} \ No newline at end of file diff --git a/unimarc/unimarc/SearchModel/KwangjuCityEduLibrarySearcher.cs b/unimarc/unimarc/SearchModel/KwangjuCityEduLibrarySearcher.cs index 9b7716d..fb4bc83 100644 --- a/unimarc/unimarc/SearchModel/KwangjuCityEduLibrarySearcher.cs +++ b/unimarc/unimarc/SearchModel/KwangjuCityEduLibrarySearcher.cs @@ -17,7 +17,7 @@ namespace BokBonCheck { public int No { get; set; } - protected string AreaCode = ""; + public string AreaCode { get; set; } = string.Empty; public string SiteName { get; protected set; } = "광주시교육청통합도서관"; public string SiteUrl => "https://lib.gen.go.kr/main/site/search/bookSearch.do#simple"; diff --git a/unimarc/unimarc/SearchModel/MokpoLibSearcher.cs b/unimarc/unimarc/SearchModel/MokpoLibSearcher.cs index 121dc54..e1b6497 100644 --- a/unimarc/unimarc/SearchModel/MokpoLibSearcher.cs +++ b/unimarc/unimarc/SearchModel/MokpoLibSearcher.cs @@ -10,7 +10,7 @@ namespace BokBonCheck { public class MokpoLibSearcher : ILibrarySearcher { - protected string AreaCode { get; set; } = string.Empty; + public string AreaCode { get; set; } = string.Empty; public string SiteName { get; protected set; } public string SiteUrl => "https://mokpolib.or.kr/dls_lt/index.php"; public bool HttpApiMode { get; set; } = true; diff --git a/unimarc/unimarc/SearchModel/MuanLibSearcher.cs b/unimarc/unimarc/SearchModel/MuanLibSearcher.cs new file mode 100644 index 0000000..8b81f1f --- /dev/null +++ b/unimarc/unimarc/SearchModel/MuanLibSearcher.cs @@ -0,0 +1,416 @@ +using System; +using System.Threading.Tasks; +using System.Text.RegularExpressions; +using OpenQA.Selenium; +using OpenQA.Selenium.Support.UI; +using System.Threading; +using OpenQA.Selenium.Chromium; +using UniMarc.SearchModel; + +namespace BokBonCheck +{ + public class MuanLibSearcher : ILibrarySearcher + { + public string AreaCode { get; set; } = string.Empty; + public string SiteName { get; protected set; } + public string SiteUrl => "https://lib.muan.go.kr/BookSearch/detail"; + public bool HttpApiMode { get; set; } = false; + + public int No { get; set; } + + private ChromiumDriver _driver; + + public MuanLibSearcher(int no, string areaCode, string areaName) + { + this.No = no; + this.AreaCode = areaCode; + this.SiteName = $"무안군립({areaName})"; + } + + public async Task StartDriver(bool showdriver = false) + { + if (_driver == null) + { + try + { + if (SeleniumHelper.IsReady == false) await SeleniumHelper.Download(); + _driver = await SeleniumHelper.CreateDriver(ShowBrowser: showdriver); + Console.WriteLine("MuanLibSearcher Driver 초기화 완료"); + } + catch (Exception ex) + { + Console.WriteLine($"MuanLibSearcher Driver 초기화 실패: {ex.Message}"); + throw new InvalidOperationException($"MuanLibSearcher Driver 초기화에 실패했습니다: {ex.Message}", ex); + } + } + } + + public void StopDriver() + { + if (_driver != null) + { + _driver.Quit(); + _driver.Dispose(); + _driver = null; + } + } + + public async Task 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)); + + // 완전한 페이지 로딩 대기 + await WaitForCompletePageLoad(wait); + + // 페이지 로드 후 브라우저 배율을 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}"); + } + + // 검색어 입력 + try + { + var searchInput = wait.Until(d => d.FindElement(By.Id("queryTitle"))); + + // 요소가 보이도록 스크롤 + ((IJavaScriptExecutor)_driver).ExecuteScript("arguments[0].scrollIntoView(true);", searchInput); + Thread.Sleep(300); + + // 기존 값 제거 + searchInput.Clear(); + + // 포커스 설정 + //searchInput.Click(); + //Thread.Sleep(200); + + // 검색어 입력 (여러 방법 시도) + try + { + searchInput.SendKeys(searchTerm); + } + catch + { + // SendKeys 실패시 JavaScript로 직접 값 설정 + ((IJavaScriptExecutor)_driver).ExecuteScript("arguments[0].value = arguments[1];", searchInput, searchTerm); + } + + // JavaScript 이벤트 발생시키기 + ((IJavaScriptExecutor)_driver).ExecuteScript("arguments[0].dispatchEvent(new Event('input', { bubbles: true }));", searchInput); + ((IJavaScriptExecutor)_driver).ExecuteScript("arguments[0].dispatchEvent(new Event('change', { bubbles: true }));", searchInput); + + Thread.Sleep(200); + + // 입력된 값 확인 + var inputValue = searchInput.GetAttribute("value"); + Console.WriteLine($"검색어 '{searchTerm}' 입력 완료, 실제 값: '{inputValue}'"); + + if (string.IsNullOrEmpty(inputValue) || !inputValue.Equals(searchTerm)) + { + Console.WriteLine("검색어 입력 재시도..."); + ((IJavaScriptExecutor)_driver).ExecuteScript("arguments[0].value = arguments[1];", searchInput, searchTerm); + ((IJavaScriptExecutor)_driver).ExecuteScript("arguments[0].dispatchEvent(new Event('input', { bubbles: true }));", searchInput); + Thread.Sleep(300); + } + } + catch (Exception ex) + { + result.ErrorMessage = $"검색어입력실패({ex.Message})"; + result.BookCount = -1; + result.IsSuccess = false; + return result; + } + + // 검색 버튼 클릭 + try + { + var searchButton = wait.Until(d => d.FindElement(By.CssSelector("button.btn.btn-lg[type='submit']"))); + + // 버튼이 보이도록 스크롤 + //((IJavaScriptExecutor)_driver).ExecuteScript("arguments[0].scrollIntoView(true);", searchButton); + //Thread.Sleep(300); + + Console.WriteLine("검색 버튼 클릭 시도..."); + + // 검색 버튼 클릭 (여러 방법 시도) + try + { + searchButton.Click(); + Console.WriteLine("일반 클릭 성공"); + } + catch + { + try + { + // JavaScript 클릭 시도 + ((IJavaScriptExecutor)_driver).ExecuteScript("arguments[0].click();", searchButton); + Console.WriteLine("JavaScript 클릭 성공"); + } + catch + { + // 폼 제출로 시도 + var form = _driver.FindElement(By.TagName("form")); + ((IJavaScriptExecutor)_driver).ExecuteScript("arguments[0].submit();", form); + Console.WriteLine("폼 제출 성공"); + } + } + + Console.WriteLine("검색 실행 완료"); + } + catch (Exception ex) + { + result.ErrorMessage = $"검색버튼클릭실패({ex.Message})"; + result.BookCount = -1; + result.IsSuccess = false; + return result; + } + + // 페이지 변경을 감지하는 메서드 + 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; + + try + { + // 검색결과 페이지 대기 + var wait = new WebDriverWait(driver, TimeSpan.FromSeconds(10)); + + // 1. 검색결과가 없는 경우 확인 + try + { + var noResultElements = driver.FindElements(By.XPath("//*[contains(text(), '검색결과가 없습니다') or contains(text(), '검색된 자료가 없습니다')]")); + if (noResultElements.Count > 0) + { + errmessage = "검색결과없음"; + return 0; + } + } + catch + { + // 검색결과가 있는 경우로 진행 + } + + // 2. totalCount에서 결과 수량 추출 + try + { + var totalCountElement = wait.Until(d => d.FindElement(By.CssSelector("p.totalCount strong"))); + if (totalCountElement != null) + { + var countText = totalCountElement.Text.Trim(); + Console.WriteLine($"totalCount 텍스트: '{countText}'"); + + if (int.TryParse(countText, out int count)) + { + if (count == 0) + { + errmessage = "검색결과없음"; + Console.WriteLine("검색 결과: 0건"); + return 0; + } + errmessage = $"검색성공({count}권)"; + Console.WriteLine($"검색 결과: {count}건"); + return count; + } + } + } + catch (Exception ex1) + { + Console.WriteLine($"totalCount 요소 추출 실패: {ex1.Message}"); + } + + // 3. 페이지 소스에서 결과 추출 + var pageSource = driver.PageSource; + Console.WriteLine("페이지 소스에서 결과 추출 시도"); + + // 검색 결과가 없다는 메시지 확인 + if (pageSource.Contains("검색결과가 없습니다") || + pageSource.Contains("검색된 자료가 없습니다") || + pageSource.Contains("자료가 없습니다") || + pageSource.Contains("전체 0 건")) + { + errmessage = "검색결과없음"; + return 0; + } + + // HTML에서 다양한 패턴 찾기 + var htmlPatterns = new[] + { + @"전체\s*\s*(\d+)\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; + } + } + } + + errmessage = "결과수량을찾을수없음"; + return -1; + + } + catch (Exception ex) + { + errmessage = ex.Message; + return -1; + } + } + + // 완전한 페이지 로딩 대기 메서드 + private async Task WaitForCompletePageLoad(WebDriverWait wait) + { + try + { + Console.WriteLine("완전한 페이지 로딩 대기 시작..."); + + // 1. document.readyState가 'complete'가 될 때까지 대기 + wait.Until(d => + { + var readyState = ((IJavaScriptExecutor)d).ExecuteScript("return document.readyState"); + return readyState.Equals("complete"); + }); + + Console.WriteLine("document.readyState = complete"); + + // 2. jQuery가 로드되고 ready 상태까지 대기 (만약 사용한다면) + try + { + wait.Until(d => + { + var jqueryReady = ((IJavaScriptExecutor)d).ExecuteScript("return typeof jQuery !== 'undefined' && jQuery.active == 0"); + return jqueryReady.Equals(true); + }); + Console.WriteLine("jQuery ready 완료"); + } + catch + { + Console.WriteLine("jQuery 없음 또는 대기 생략"); + } + + // 3. 추가 대기 시간 + await Task.Delay(10); + + // 4. 검색 입력창이 실제로 존재하고 상호작용 가능할 때까지 대기 + //wait.Until(d => + //{ + // try + // { + // var searchInput = d.FindElement(By.Id("queryTitle")); + // return searchInput != null && searchInput.Displayed && searchInput.Enabled; + // } + // catch + // { + // return false; + // } + //}); + + //Console.WriteLine("검색 입력창 준비 완료"); + + } + catch (Exception ex) + { + Console.WriteLine($"페이지 로딩 대기 중 오류: {ex.Message}"); + // 오류가 발생해도 최소한의 대기 시간 적용 + await Task.Delay(3000); + } + } + + public async Task WaitForPageChange(WebDriverWait wait) + { + try + { + // 방법 4: 페이지 로딩 상태 확인 + wait.Until(d => + { + var readyState = ((IJavaScriptExecutor)d).ExecuteScript("return document.readyState"); + return readyState.Equals("complete"); + }); + + // 방법 5: 특정 텍스트가 페이지에 나타날 때까지 대기 + wait.Until(d => + { + var elm = d.FindElement(By.TagName("body")); + if (elm == null) return false; + var pageText = elm.Text; + return pageText.Contains("전체") || pageText.Contains("건") || pageText.Contains("검색결과"); + }); + } + catch (Exception ex) + { + // 모든 감지 방법이 실패하면 최소한의 대기 시간 적용 + await Task.Delay(2000); + throw new Exception($"페이지 변경 감지 실패: {ex.Message}"); + } + } + } +} \ No newline at end of file diff --git a/unimarc/unimarc/SearchModel/NamguLibrarySearcher.cs b/unimarc/unimarc/SearchModel/NamguLibrarySearcher.cs index b4b462c..80e3a90 100644 --- a/unimarc/unimarc/SearchModel/NamguLibrarySearcher.cs +++ b/unimarc/unimarc/SearchModel/NamguLibrarySearcher.cs @@ -21,7 +21,7 @@ namespace BokBonCheck public class NamguLibrarySearcher : ILibrarySearcher { - protected string AreaCode = ""; + public string AreaCode { get; set; } = string.Empty; public string SiteName { get; protected set; } = "남구통합도서관(전체)"; public string SiteUrl => "https://lib.namgu.gwangju.kr/main/bookSearch"; public bool HttpApiMode { get; set; } = false; diff --git a/unimarc/unimarc/SearchModel/SuncheonLibSearcher.cs b/unimarc/unimarc/SearchModel/SuncheonLibSearcher.cs index f6bc42b..91d571a 100644 --- a/unimarc/unimarc/SearchModel/SuncheonLibSearcher.cs +++ b/unimarc/unimarc/SearchModel/SuncheonLibSearcher.cs @@ -10,7 +10,7 @@ namespace BokBonCheck { public class SuncheonLibSearcher : ILibrarySearcher { - protected string AreaCode { get; set; } = string.Empty; + public string AreaCode { get; set; } = string.Empty; public string SiteName { get; protected set; } public string SiteUrl => "https://library.suncheon.go.kr/lib/book/search/searchIndex.do"; public bool HttpApiMode { get; set; } = true; diff --git a/unimarc/unimarc/SearchModel/WandoLibSearcher.cs b/unimarc/unimarc/SearchModel/WandoLibSearcher.cs index e1eb1f0..5d79869 100644 --- a/unimarc/unimarc/SearchModel/WandoLibSearcher.cs +++ b/unimarc/unimarc/SearchModel/WandoLibSearcher.cs @@ -20,7 +20,7 @@ namespace BokBonCheck { public class WandoLibSearcher : ILibrarySearcher { - protected string AreaCode { get; set; } = string.Empty; + public string AreaCode { get; set; } = string.Empty; public string SiteName { get; protected set; } public string SiteUrl => "https://wandolib.kr/dls_le/index.php"; public bool HttpApiMode { get; set; } = true; diff --git a/unimarc/unimarc/SearchModel/YeosuLibSearcher.cs b/unimarc/unimarc/SearchModel/YeosuLibSearcher.cs index d403b32..956aa95 100644 --- a/unimarc/unimarc/SearchModel/YeosuLibSearcher.cs +++ b/unimarc/unimarc/SearchModel/YeosuLibSearcher.cs @@ -12,7 +12,7 @@ namespace BokBonCheck public class YeosuLibSearcher : ILibrarySearcher { - protected string AreaCode { get; set; } = string.Empty; + public string AreaCode { get; set; } = string.Empty; public string SiteName { get; protected set; } public string SiteUrl => "https://yslib.yeosu.go.kr/dls_kapi/index.php"; diff --git a/unimarc/unimarc/UniMarc.csproj b/unimarc/unimarc/UniMarc.csproj index cd5ce03..271bf07 100644 --- a/unimarc/unimarc/UniMarc.csproj +++ b/unimarc/unimarc/UniMarc.csproj @@ -230,6 +230,8 @@ + + Form @@ -245,7 +247,9 @@ + + diff --git a/unimarc/unimarc/마크/Check_copyWD.cs b/unimarc/unimarc/마크/Check_copyWD.cs index d413b7b..864b84e 100644 --- a/unimarc/unimarc/마크/Check_copyWD.cs +++ b/unimarc/unimarc/마크/Check_copyWD.cs @@ -375,6 +375,55 @@ namespace WindowsFormsApp1.Mac idx = 1580; _searchService.AddSearcher(new GyeongnamLibSearcher(idx++, "", "경남대표도서관")); + // 무안군립도서관 + idx = 1590; + _searchService.AddSearcher(new MuanLibSearcher(idx++, "", "무안군립도서관")); + + // 조선대학교중앙도서관 + idx = 1600; + _searchService.AddSearcher(new ChosunUnivLibSearcher(idx++, "", "조선대학교중앙도서관")); + + // 조선이공대학교도서관 + idx = 1610; + _searchService.AddSearcher(new ChosunTechLibSearcher(idx++, "", "조선이공대학교도서관")); + + // KCM통합도서관 (여수시) + idx = 1620; + _searchService.AddSearcher(new KcmLibSearcher(idx++, "MA", "쌍봉도서관")); + _searchService.AddSearcher(new KcmLibSearcher(idx++, "MG", "여수이순신도서관")); + _searchService.AddSearcher(new KcmLibSearcher(idx++, "MB", "여수시립현암도서관")); + _searchService.AddSearcher(new KcmLibSearcher(idx++, "MC", "여수시립환경도서관")); + _searchService.AddSearcher(new KcmLibSearcher(idx++, "MD", "여수시립돌산도서관")); + _searchService.AddSearcher(new KcmLibSearcher(idx++, "ME", "여수시립소라도서관")); + _searchService.AddSearcher(new KcmLibSearcher(idx++, "MF", "여수시립율촌도서관")); + _searchService.AddSearcher(new KcmLibSearcher(idx++, "PA", "거문도은빛바다도서관")); + _searchService.AddSearcher(new KcmLibSearcher(idx++, "PB", "치매안심작은도서관")); + _searchService.AddSearcher(new KcmLibSearcher(idx++, "PC", "청솔글누리작은도서관")); + _searchService.AddSearcher(new KcmLibSearcher(idx++, "PD", "동부도시보건작은도서관")); + _searchService.AddSearcher(new KcmLibSearcher(idx++, "PE", "화양열린작은도서관")); + _searchService.AddSearcher(new KcmLibSearcher(idx++, "PF", "여문늘벗작은도서관")); + _searchService.AddSearcher(new KcmLibSearcher(idx++, "PH", "국동협동관")); + _searchService.AddSearcher(new KcmLibSearcher(idx++, "SA", "아주타운아파트작은도서관")); + _searchService.AddSearcher(new KcmLibSearcher(idx++, "SB", "책이랑나랑작은도서관")); + _searchService.AddSearcher(new KcmLibSearcher(idx++, "SC", "현천작은도서관")); + _searchService.AddSearcher(new KcmLibSearcher(idx++, "SE", "꿈꾸는영어전문작은도서관")); + _searchService.AddSearcher(new KcmLibSearcher(idx++, "SF", "학마을작은도서관")); + _searchService.AddSearcher(new KcmLibSearcher(idx++, "SG", "웅천지웰작은도서관")); + _searchService.AddSearcher(new KcmLibSearcher(idx++, "SH", "한려작은도서관")); + _searchService.AddSearcher(new KcmLibSearcher(idx++, "SJ", "주은금호작은도서관")); + _searchService.AddSearcher(new KcmLibSearcher(idx++, "SK", "광림작은도서관")); + _searchService.AddSearcher(new KcmLibSearcher(idx++, "SM", "민들레작은도서관")); + _searchService.AddSearcher(new KcmLibSearcher(idx++, "SO", "원앙작은도서관")); + _searchService.AddSearcher(new KcmLibSearcher(idx++, "SP", "푸른정원작은도서관")); + _searchService.AddSearcher(new KcmLibSearcher(idx++, "SR", "로얄골드빌작은도서관")); + _searchService.AddSearcher(new KcmLibSearcher(idx++, "SS", "꿈을키우는작은도서관")); + _searchService.AddSearcher(new KcmLibSearcher(idx++, "SY", "신기부영작은도서관")); + _searchService.AddSearcher(new KcmLibSearcher(idx++, "SZ", "국동365열린도서관")); + _searchService.AddSearcher(new KcmLibSearcher(idx++, "TA", "지웰2차작은도서관")); + _searchService.AddSearcher(new KcmLibSearcher(idx++, "TB", "이편한작은도서관")); + _searchService.AddSearcher(new KcmLibSearcher(idx++, "TC", "채움늘작은도서관")); + _searchService.AddSearcher(new KcmLibSearcher(idx++, "TD", "웅천글꽃작은도서관")); + } this.tb_SearchTarget.Items.Clear(); @@ -869,7 +918,7 @@ namespace WindowsFormsApp1.Mac return; } chkShowBrowser.Enabled = !searcher.HttpApiMode; - this.lbSite.Text = searcher.SiteUrl; + this.lbSite.Text = $"[{searcher.AreaCode}] {searcher.SiteUrl}"; } } }