242 lines
6.9 KiB
Markdown
242 lines
6.9 KiB
Markdown
# 수인약품 (Sooin) API 문서
|
|
|
|
## 개요
|
|
- **회사명**: 수인약품
|
|
- **웹사이트**: http://sooinpharm.co.kr
|
|
- **인증방식**: Playwright 로그인 → requests 쿠키 재사용 (세션 30분 유효)
|
|
|
|
## 인증 정보 (환경변수)
|
|
```
|
|
SOOIN_USER_ID=thug0bin
|
|
SOOIN_PASSWORD=@Trajet6640
|
|
SOOIN_VENDOR_CODE=50911
|
|
SOOIN_VENDOR_NAME=청춘약국
|
|
```
|
|
|
|
## 핵심 URL 구조
|
|
|
|
### 기존 구현된 URL
|
|
| URL | 용도 |
|
|
|-----|------|
|
|
| `/Homepage/intro.asp` | 로그인 페이지 |
|
|
| `/Service/Order/Order.asp` | 제품 검색 |
|
|
| `/Service/Order/BagOrder.asp` | 장바구니 추가 |
|
|
| `/Service/Order/Bag.asp` | 장바구니 조회 |
|
|
| `/Service/Order/ControlBag.asp` | 장바구니 항목 제어 (취소/복원) |
|
|
| `/Service/Order/OrderEnd.asp` | 주문 전송 (확정) |
|
|
| `/Service/Order/PhysicInfo.asp` | 제품 상세 정보 |
|
|
| `/Service/SalesLedger/SalesLedger.asp` | 잔고/매출 조회 |
|
|
|
|
### 🆕 신규 구현 필요 URL (주문 조회)
|
|
| URL | 용도 |
|
|
|-----|------|
|
|
| `/Service/Report/Report.asp` | **주문 내역 목록 조회** |
|
|
| `/Service/Report/Report.asp?f=view&orderNum=...` | **주문 상세 조회** |
|
|
|
|
## 주문 조회 URL 분석
|
|
|
|
### 주문 목록 URL 예시
|
|
```
|
|
http://sooinpharm.co.kr/Service/Report/Report.asp?
|
|
sDate=2026-03-01 # 시작일
|
|
&eDate=2026-03-07 # 종료일
|
|
&tx_ven=50911 # 거래처 코드 (SOOIN_VENDOR_CODE)
|
|
&currVenNm=청춘약국 # 거래처명 (URL 인코딩됨)
|
|
&orderStatus=0 # 주문상태 (0=전체?)
|
|
&ListOrder=0 # 정렬 (0=기본)
|
|
&PhysicNm= # 제품명 필터 (선택)
|
|
&sg= # 미확인
|
|
&page=1 # 페이지 번호
|
|
```
|
|
|
|
### 주문 상세 URL 예시
|
|
```
|
|
http://sooinpharm.co.kr/Service/Report/Report.asp?
|
|
f=view # view 모드 (상세 조회)
|
|
&orderNum=202603095091177 # 주문번호 (YYYYMMDD + 거래처코드 + 순번?)
|
|
&Ifflag= # 미확인
|
|
&sDate=2026-03-01
|
|
&eDate=2026-03-07
|
|
&PhysicNm=
|
|
&ListOrder=0
|
|
&tx_ven=50911
|
|
&currVenNm=청춘약국
|
|
&orderStatus=0
|
|
&sg=
|
|
&page=1
|
|
```
|
|
|
|
### 주문번호 구조 분석
|
|
- `202603095091177` → `20260309` (날짜) + `50911` (거래처코드) + `77` (순번?)
|
|
- 형식: `YYYYMMDD` + `VENDOR_CODE` + `SEQ`
|
|
|
|
## API 엔드포인트
|
|
|
|
### Flask Blueprint: `/api/sooin/*`
|
|
|
|
| Method | Endpoint | 설명 |
|
|
|--------|----------|------|
|
|
| GET | `/api/sooin/stock` | 재고 조회 |
|
|
| GET | `/api/sooin/session-status` | 세션 상태 확인 |
|
|
| GET | `/api/sooin/balance` | 잔고(미수금) 조회 |
|
|
| GET | `/api/sooin/monthly-sales` | 월간 매출 조회 |
|
|
| GET | `/api/sooin/cart` | 장바구니 조회 |
|
|
| POST | `/api/sooin/cart/clear` | 장바구니 비우기 |
|
|
| POST | `/api/sooin/cart/cancel` | 장바구니 항목 취소 |
|
|
| POST | `/api/sooin/cart/restore` | 취소 항목 복원 |
|
|
| POST | `/api/sooin/order` | 주문 (장바구니 추가) |
|
|
| POST | `/api/sooin/confirm` | 주문 확정 |
|
|
| POST | `/api/sooin/full-order` | 전체 주문 (검색→담기→확정) |
|
|
| POST | `/api/sooin/order-batch` | 일괄 주문 |
|
|
| **GET** | **`/api/sooin/orders`** | **✅ 주문 목록 조회** |
|
|
| **GET** | **`/api/sooin/orders/<order_num>`** | **✅ 주문 상세 조회** |
|
|
| **GET** | **`/api/sooin/orders/today-summary`** | **✅ 오늘 주문 집계** |
|
|
|
|
## 🆕 신규 구현 목표: 주문 조회 API
|
|
|
|
### 목적
|
|
1. **오늘 주문한 약 목록 조회** - 주문이 정상적으로 들어갔는지 확인
|
|
2. **제품별 주문량 집계** - 오늘의 총 주문량을 제품별로 파악
|
|
|
|
### 구현할 API 엔드포인트
|
|
|
|
#### 1. 주문 목록 조회
|
|
```
|
|
GET /api/sooin/orders?start_date=2026-03-01&end_date=2026-03-07
|
|
```
|
|
**응답 예시:**
|
|
```json
|
|
{
|
|
"success": true,
|
|
"orders": [
|
|
{
|
|
"order_num": "202603095091177",
|
|
"order_date": "2026-03-09",
|
|
"order_time": "14:30:25",
|
|
"total_amount": 125000,
|
|
"item_count": 5,
|
|
"status": "완료"
|
|
}
|
|
],
|
|
"total_count": 10
|
|
}
|
|
```
|
|
|
|
#### 2. 주문 상세 조회
|
|
```
|
|
GET /api/sooin/orders/<order_num>
|
|
```
|
|
**응답 예시:**
|
|
```json
|
|
{
|
|
"success": true,
|
|
"order_num": "202603095091177",
|
|
"order_date": "2026-03-09",
|
|
"items": [
|
|
{
|
|
"product_code": "32495", // 수인 내부코드
|
|
"kd_code": "073100220", // KD코드 (있으면)
|
|
"product_name": "코자정50mg",
|
|
"spec": "30T",
|
|
"quantity": 2,
|
|
"unit_price": 15000,
|
|
"amount": 30000
|
|
}
|
|
],
|
|
"total_amount": 125000
|
|
}
|
|
```
|
|
|
|
#### 3. 오늘 주문 집계 (제품별)
|
|
```
|
|
GET /api/sooin/orders/today-summary
|
|
```
|
|
**응답 예시:**
|
|
```json
|
|
{
|
|
"success": true,
|
|
"date": "2026-03-09",
|
|
"summary": [
|
|
{
|
|
"kd_code": "073100220",
|
|
"product_name": "코자정50mg",
|
|
"total_quantity": 10,
|
|
"total_amount": 150000,
|
|
"order_count": 3
|
|
}
|
|
],
|
|
"grand_total_amount": 500000,
|
|
"grand_total_items": 25
|
|
}
|
|
```
|
|
|
|
## 파일 구조
|
|
|
|
```
|
|
pharmacy-wholesale-api/
|
|
├── wholesale/
|
|
│ ├── __init__.py
|
|
│ ├── base.py # 공통 베이스 클래스
|
|
│ ├── sooin.py # 수인약품 핵심 로직 ← 여기에 주문 조회 메서드 추가
|
|
│ ├── geoyoung.py
|
|
│ └── baekje.py
|
|
└── .env # 인증 정보
|
|
|
|
pharmacy-pos-qr-system/
|
|
├── backend/
|
|
│ ├── app.py # 메인 Flask 앱
|
|
│ ├── sooin_api.py # 수인약품 Flask Blueprint ← 여기에 API 엔드포인트 추가
|
|
│ ├── wholesale_path.py # wholesale 패키지 경로 설정
|
|
│ └── templates/
|
|
│ └── admin_rx_usage.html # 프론트엔드 (주문 조회 UI 추가 가능)
|
|
└── docs/
|
|
└── SOOIN_API.md # 이 문서
|
|
```
|
|
|
|
## 구현 가이드
|
|
|
|
### 1단계: SooinSession에 메서드 추가 (`wholesale/sooin.py`)
|
|
|
|
```python
|
|
def get_order_list(self, start_date: str, end_date: str) -> dict:
|
|
"""주문 목록 조회"""
|
|
# /Service/Report/Report.asp 파싱
|
|
pass
|
|
|
|
def get_order_detail(self, order_num: str) -> dict:
|
|
"""주문 상세 조회"""
|
|
# /Service/Report/Report.asp?f=view&orderNum=... 파싱
|
|
pass
|
|
|
|
def get_today_order_summary(self) -> dict:
|
|
"""오늘 주문 제품별 집계"""
|
|
# get_order_list + get_order_detail 조합
|
|
pass
|
|
```
|
|
|
|
### 2단계: Flask API 엔드포인트 추가 (`sooin_api.py`)
|
|
|
|
```python
|
|
@sooin_bp.route('/orders', methods=['GET'])
|
|
def api_sooin_orders():
|
|
"""주문 목록 조회"""
|
|
pass
|
|
|
|
@sooin_bp.route('/orders/<order_num>', methods=['GET'])
|
|
def api_sooin_order_detail(order_num):
|
|
"""주문 상세 조회"""
|
|
pass
|
|
|
|
@sooin_bp.route('/orders/today-summary', methods=['GET'])
|
|
def api_sooin_today_summary():
|
|
"""오늘 주문 집계"""
|
|
pass
|
|
```
|
|
|
|
## 주의사항
|
|
|
|
1. **인코딩**: 수인약품 사이트는 `euc-kr` 인코딩 사용
|
|
2. **세션 유지**: 30분 세션 타임아웃, 자동 재로그인 필요
|
|
3. **HTML 파싱**: BeautifulSoup으로 테이블 구조 파싱
|
|
4. **에러 처리**: 로그인 실패, 네트워크 오류 등 처리 필요
|