Skip to content

Commit

Permalink
[Scrap] 결과 이메일로 전송되도록 작성
Browse files Browse the repository at this point in the history
  • Loading branch information
Re-st committed Nov 21, 2023
1 parent 6f2f9f8 commit 19d98ce
Show file tree
Hide file tree
Showing 5 changed files with 191 additions and 58 deletions.
9 changes: 9 additions & 0 deletions configurations/secrets.py
Original file line number Diff line number Diff line change
Expand Up @@ -27,3 +27,12 @@ class OpenDataPortalSecrets:
"""

service_key = str(os.getenv("OPEN_DATA_SERICE_KEY") or "")

class EmailSecrets:
"""
스크랩 결과 이메일 전송에 필요한 키를 정의합니다.
"""

sender_email = str(os.getenv("SCRAP_SENDER_EMAIL") or "")
receiver_email = str(os.getenv("SCRAP_RECEIVER_EMAIL") or "")
password = str(os.getenv("SCRAP_EMAIL_PASSWORD") or "")
115 changes: 115 additions & 0 deletions scrap/local_councils/daejeon.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,115 @@
from scrap.local_councils import *


def scrap_65(url, cid) -> ScrapResult:
"""대전 동구"""
soup = get_soup(url, verify=False)
councilors: List[Councilor] = []

# 프로필 링크 스크랩을 위해 base_url 추출
parsed_url = urlparse(url)
base_url = f"{parsed_url.scheme}://{parsed_url.netloc}"

for profile in soup.find_all("dl", class_="profile"):
name_tag = profile.find("strong", class_="name")
name = name_tag.get_text(strip=True) if name_tag else "이름 정보 없음"
party = "정당 정보 없음"

# 프로필보기 링크 가져오기
profile_link = profile.find("a", class_="start")
if profile_link:
data_uid = profile_link.get("data-uid")
if data_uid:
profile_url = base_url + f"/kr/member/profile_popup?uid={data_uid}"
profile_soup = get_soup(profile_url, verify=False)
party_info = profile_soup.find("strong", string="정 당")
if (
party_info
and (party_span := party_info.find_next("span")) is not None
):
party = party_span.text

councilors.append(Councilor(name=name, jdName=party))

return ret_local_councilors(cid, councilors)


def scrap_66(url, cid) -> ScrapResult:
"""대전 중구"""
soup = get_soup(url, verify=False)
councilors: List[Councilor] = []

for profile in soup.find_all("div", class_="profile"):
name_tag = profile.find("div", class_="name")
name = name_tag.get_text(strip=True) if name_tag else "이름 정보 없음"

party = "정당 정보 없음"
party_info = profile.find("em", string="소속정당")
if party_info:
party = party_info.find_next("span").get_text(strip=True)
councilors.append(Councilor(name=name, jdName=party))

return ret_local_councilors(cid, councilors)


def scrap_67(
url,
cid,
) -> ScrapResult:
"""대전 서구"""
soup = get_soup(url, verify=False)
councilors: List[Councilor] = []

for profile in soup.find_all("dl"):
name_tag = profile.find("dd", class_="name")
name = (
name_tag.get_text(strip=True).replace(" 의원", "") if name_tag else "이름 정보 없음"
)

party = "정당 정보 없음"
party_info = list(filter(lambda x: "정당" in str(x), profile.find_all("dd")))
if party_info:
party = party_info[0].get_text(strip=True).replace("정당: ", "")

councilors.append(Councilor(name=name, jdName=party))

return ret_local_councilors(cid, councilors)


def scrap_68(url, cid) -> ScrapResult:
"""대전 유성구"""
soup = get_soup(url, verify=False)
councilors: List[Councilor] = []

for profile in soup.find_all("div", class_="profile"):
name_tag = profile.find("em", class_="name")
# () 안에 있는 한자를 제거 (ex. 김영희(金英姬) -> 김영희)
name = name_tag.get_text(strip=True).split("(")[0] if name_tag else "이름 정보 없음"

party = "정당 정보 없음"
regex_pattern = re.compile(r"정\s*당\s*:", re.IGNORECASE) # Case-insensitive
party_info = profile.find("em", string=regex_pattern)
if party_info:
party = party_info.find_next("span").get_text(strip=True)
councilors.append(Councilor(name=name, jdName=party))

return ret_local_councilors(cid, councilors)


def scrap_69(url, cid) -> ScrapResult:
"""대전 대덕구"""
soup = get_soup(url, verify=False)
councilors: List[Councilor] = []

for profile in soup.find_all("div", class_="profile"):
name_tag = profile.find("em", class_="name")
name = name_tag.get_text(strip=True) if name_tag else "이름 정보 없음"

party = "정당 정보 없음"
regex_pattern = re.compile(r"정\s*당\s*:", re.IGNORECASE) # Case-insensitive
party_info = profile.find("em", string=regex_pattern)
if party_info:
party = party_info.find_next("span").get_text(strip=True)
councilors.append(Councilor(name=name, jdName=party))

return ret_local_councilors(cid, councilors)
4 changes: 4 additions & 0 deletions scrap/utils/__init__.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,4 @@
"""
크롤링을 실행, 진행결과 알림, 크롤링결과를 mongoDB로 저장하는
기능을 담당하는 모듈입니다.
"""
26 changes: 26 additions & 0 deletions scrap/utils/email_result.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,26 @@
import smtplib
from email.mime.text import MIMEText
from configurations.secrets import EmailSecrets

# 이메일 서버 설정 (Gmail 사용 예제)
smtp_server = "smtp.gmail.com"
smtp_port = 587

def email_result(emessages):
# 이메일 내용 설정
subject = "스크래핑 결과"
# 메일 구성
msg = MIMEText(emessages)
msg['Subject'] = subject
msg['From'] = EmailSecrets.sender_email
msg['To'] = EmailSecrets.receiver_email

# 이메일 전송
try:
with smtplib.SMTP(smtp_server, smtp_port) as server:
server.starttls()
server.login(msg['From'], EmailSecrets.password)
server.sendmail(msg['From'], msg['To'], msg.as_string())
print("이메일이 성공적으로 전송되었습니다.")
except Exception as e:
print(f"이메일 전송 중 오류 발생: {e}")
95 changes: 37 additions & 58 deletions scrap/utils/spreadsheet.py
Original file line number Diff line number Diff line change
@@ -1,3 +1,6 @@
"""
local_councils 폴더에 정의된 각 함수를 사용해서 크롤링합니다.
"""
import os
import sys
import gspread
Expand All @@ -20,6 +23,7 @@
from scrap.local_councils.gyeongsang import *
from scrap.local_councils import *
from requests.exceptions import Timeout
from utils.email_result import email_result

# 구글로부터 권한을 요청할 어플리케이션 목록
# 변경 시 token.json 삭제 후 재인증 필요
Expand Down Expand Up @@ -67,60 +71,22 @@ def main() -> None:
0
) # 원하는 워크시트 선택 (0은 첫 번째 워크시트입니다.)
# TODO - 홈페이지 위 charset=euc-kr 등을 인식해 바로 가져오기.
euc_kr = [
6,
13,
16,
31,
72,
88,
112,
134,
154,
157,
163,
165,
167,
176,
181,
197,
202,
222,
]
euc_kr = [6, 13, 16, 31, 72, 88, 112, 134, 154, 157, 163, 165, 167, 176, 181,
197, 202, 222]
special_functions = (
list(range(1, 57))
+ [62, 63, 64, 88, 97, 103, 107]
+ list(range(113, 127))
+ [132, 134, 140, 142, 154, 155, 156, 157, 160, 161, 162, 163, 164, 165, 167]
+ list(range(177, 180))
+ [
182,
183,
184,
186,
188,
189,
190,
191,
194,
195,
196,
198,
199,
201,
203,
206,
208,
209,
210,
]
+ [182, 183, 184, 186, 188, 189, 190, 191, 194, 195, 196, 198, 199, 201, 203,
206, 208, 209, 210]
+ list(range(212, 221))
+ [222, 223, 224, 226]
)
selenium_basic = [76, 78, 101, 169, 173, 177]
no_information = [18, 29, 106, 111, 172, 181, 185, 187, 197, 200, 204, 207]
error_unsolved = [170, 171]
errors = []
f = open(JSON_PATH, "r")
args = json.load(f)
f.close()
Expand All @@ -132,16 +98,22 @@ def main() -> None:
parse_error_times = 0
timeouts = 0
N = 226
for n in [189]: # range(1, N + 1):
emessages: str = ""
enumbers = []
def add_error(n, msg):
nonlocal emessages
emsg: str = f"| {n:3} | 오류: {msg}"
emessages += emsg
enumbers.append(n)
for n in range(1, N + 1):
if n in no_information + error_unsolved:
error_msg = (
emsg: str = (
"지난번 확인 시, 정당 정보 등이 홈페이지에 없었습니다. \
다시 확인해보시겠어요?"
if n in no_information
else "함수 구현에 실패한 웹페이지입니다."
)
print(f"| {n} | 오류: ", error_msg, " 링크 : ", data[n - 1]["URL"])
errors.append(n)
) + " 링크: " + data[n - 1]["URL"]
add_error(n, emsg)
continue
encoding = "euc-kr" if n in euc_kr else "utf-8"
council_url: str = ""
Expand All @@ -160,28 +132,35 @@ def main() -> None:
function_to_call(council_url, n, args=council_args).councilors
)
else:
print("[API/spreadsheet] Error : No function found")
emsg: str = f"특수 함수를 사용해서 스크랩한다고 \
명시되어 있는데 함수가 정의되어 있지 않네요. [scrap/utils/\
spreadsheet.py의 special_functions에 함수 번호를 빼고 \
다시 시도해 보시겠어요?]"
add_error(n, emsg)
elif n in selenium_basic:
result = str(sel_scrap_basic(council_url, n, council_args).councilors)
else:
result = str(
scrap_basic(council_url, n, council_args, encoding).councilors
)
if "정보 없음" in result:
print("정보 없음이 포함되어 있습니다.")
emsg = "스크랩 결과에 '정보 없음'이 포함되어 있습니다. 일부 인명에\
대해 스크랩이 실패했다는 뜻이에요. 함수나 인자를 점검해 주세요."
parse_error_times += 1
errors.append(n)
print(f"| {n} | {result}")
add_error(n, emsg)
except Timeout:
print(f"| {n} | 오류: Request to {council_url} timed out.")
emsg = f"{council_url}에 시도한 연결이 타임아웃됐어요."
timeouts += 1
add_error(n, emsg)
except Exception as e:
print(f"| {n} | 오류: {e}")
errors.append(n)
continue # 에러가 발생하면 다음 반복으로 넘어감
print(
f"| 총 실행 횟수: {N} | 에러: {errors}, 총 {len(errors)}회 | 그 중 정보 없음 횟수: {parse_error_times} | 타임아웃 횟수: {timeouts} |"
)
add_error(n, "기타 오류 - " + str(e))
emessages = f"""
총 실행 횟수: {N}
에러: {enumbers}, 총 {len(enumbers)}
그 중 '정보 없음' 횟수: {parse_error_times}
타임아웃 횟수: {timeouts}
""" + emessages
email_result(emessages)


if __name__ == "__main__":
Expand Down

0 comments on commit 19d98ce

Please sign in to comment.