Perfect 미니 프로젝트 [3]
- 사진 게시물 등록하기
- URL을 통해 사진 GET
API를 하나씩 하나씩 완성해 나가도록 합시다!!
참고
2017년 8월 현재 Swift 3 / Perfect 최신버전 2.0.x 환경에서 진행함을 알려드립니다.
지난 내용 돌아보기
- API 구성
- 라우팅 및 DB 구성
사진 게시물 등록하기
먼저 CRUD 중, Create를 먼저 해보도록 할게요. 사진 게시물 등록을 위한 API를 구현해 봅니다. 사진 게시물 등록은 /article
경로에 POST
메서드를 통해 multipart/form-data
를 전송합니다.
지난 번 정의한 API 스펙 중 사진 게시물 등록 API의 정보입니다.
요청(Request)
- HTTP Method : POST
- Content-Type: multipart/form-data
매개변수 |
자료형 |
값의 범위/기본 값 |
비고 |
필수여부 |
image |
binary data |
|
이미지 데이터 |
Y |
user_name |
string |
|
사용자 이름 |
Y |
description |
string |
|
이미지 설명 |
N |
title |
string |
|
이미지 제목 |
Y |
응답(Response)
Key |
자료형 |
비고 |
필수여부 |
article_id |
string |
업로드된 게시물 고유 식별자 |
Y |
image_url |
string |
이미지 주소 |
Y |
user_name |
string |
사용자 이름 |
Y |
description |
string |
이미지 설명 |
N |
title |
string |
이미지 제목 |
Y |
우선 Sources/handlers.swift 파일로 이동하여 반복적으로 수행할 것 같은 기능이나, 상수를 정의해보려 합니다.
먼저 필수 매개변수가 모두 실려왔는지 확인하는 함수를 만듭니다. handlers.swift 파일 맨 아래쪽에 함수를 만들어봅니다.
// 부족한 매개변수가 없는지 확인하여 부족한 매개변수가 있다면 String Array로 반환
private func lakedParams(paramsNeeded: [String], paramsReceived: [(String, String)]) -> [String]? {
var laked: [String] = []
for param in paramsNeeded {
if paramsReceived.filter({ (key, _) -> Bool in
return key == param
}).isEmpty {
laked.append(param)
}
}
return laked.count > 0 ? laked : nil
}
또, 나중에 해보시면 아시겠지만 JSONConvertible
타입의 데이터를 응답 객체의 body에 셋팅하려면 꼭 오류처리(try)를 해줘야 합니다. 매 번 하기 귀찮아서 그냥 함수를 하나 만들어줬습니다. lakedParams(paramsNeeded:paramsReceived:)
함수 아래에 작성해봅니다.
// 응답 보내기
private func sendCompleteResponse(_ response: HTTPResponse, status: HTTPResponseStatus = .ok, jsonBody: JSONConvertible? = nil) {
do {
if let json = jsonBody {
try response.setBody(json: json)
}
response.completed(status: status)
} catch {
response.completed(status: .internalServerError)
}
}
참, 그리고 요청에서 실려온 이미지를 저장할 디렉터리도 미리 지정해 두고 싶구요, 매개변수 이름이나 JSON 키값도 미리 지정해두고 싶습니다. ContentsType
구조체 선언 아래쪽에 아래 코드를 추가해봅니다. 사실 어디에 위치하든 크게 상관은 없지만 그래도 이게 예쁘잖아요 😀
// 이미지 저장 디렉터리
private let imageDir: Dir? = {
let imageDir = Dir("./image")
if imageDir.exists == false {
do {
try imageDir.create()
print("Working Directory (\(imageDir.path)) for examples created.")
} catch {
print("Could not create Working Directory for examples.")
return nil
}
}
return imageDir
}()
// JSON 데이터의 Key 값 혹은 요청 매개변수 이름에 사용
private struct JSONKey {
static let image = "image"
static let userName = "user_name"
static let description = "description"
static let title = "title"
static let articles = "articles"
static let articleId = "article_id"
static let page = "page"
static let articlesPerPage = "articles_per_page"
static let currentPage = "current_page"
static let totalPage = "total_page"
static let imageUrl = "image_url"
static let imagePath = "image_path"
static let message = "message"
static let error = "error"
}
자, 밑준비를 마쳤습니다.
사용자가 게시물 작성을 요청하는 postArticleHandler(request: response:)
함수를 본격적으로 작성해봅시다. 기존에 함수 내부에 있던 코드는 지워주세요 🙂
먼저, 응답은 JSON 형식이 될 것이므로 맨 윗줄에 아래 코드를 작성합니다. response
객체는 클라이언트에게 응답을 줄 때 사용할 객체입니다.
// 응답 컨텐츠 형식은 JSON
response.setHeader(.contentType, value: ContentsType.json)
매개변수로 전달된 request
객체에 클라이언트의 요청 정보가 모두 실려있습니다. 먼저 모든 필수 매개변수가 전달되었는지 확인합니다.
POST
메서드로 전달된 매개변수는 request
객체의 postParams()
메서드를 통해 가져올 수 있습니다. postParams()
메서드의 반환 타입은 (String, String)
튜플의 Array
타입입니다.
필수 매개변수인 user_name
, title
이 모두 전달되었는지 확인하고 싶습니다(image
매개변수는 파일로 받아올 것이니 다음 차례에서 확인합니다).
// 부족한 매개변수가 없는지 확인
if let lakedParams = lakedParams(paramsNeeded: [JSONKey.userName, JSONKey.title], paramsReceived: request.postParams) {
sendCompleteResponse(response, status: .badRequest, jsonBody: [JSONKey.message:"need more params \(lakedParams)"])
}
// 이미지 파일이 업로드 되었는지 확인
guard let imageInformation: MimeReader.BodySpec = request.postFileUploads?.first,
let imageFile = imageInformation.file else {
sendCompleteResponse(response, status: .badRequest, jsonBody: [JSONKey.message:"need image file"])
return
}
자, 이렇게 모든 전달 데이터가 확인되었다면 이미지를 서버의 이미지 디렉터리에 저장할 차례입니다. 먼저 이미지 이름에 사용자 이름을 넣고 싶기 때문에 클라이언트에서 전달된 데이터를 빼냅니다.
// 사용자 이름, 제목 추출
guard let userName = request.param(name: JSONKey.userName),
let title = request.param(name: JSONKey.title) else {
sendCompleteResponse(response, status: .internalServerError, jsonBody: nil)
return
}
그리고 이미지가 저장될 디렉터리에 접근할 수 있는지 확인합니다.
// 이미지가 저장될 디렉터리
guard let imageDirectory = imageDir else {
sendCompleteResponse(response, status: .internalServerError, jsonBody: nil)
return
}
고유한 이미지 파일 이름을 만들어주기 위해서 타임스탬프 값을 활용했습니다. 위에서 request
객체에서 꺼내온 이미지 파일을 서버의 이미지 디렉터리 내부의 경로로 복사해줍니다.
// 고유한 이미지 이름을 위해 타임스템프 값을 활용
let timestamp: Int = icuDateToSeconds(getNow())
// 사용자이름_타임스템프.jpg 형식으로 파일이름 지정
let imageFileName: String = userName + "_" + String(timestamp) + ".jpg"
// 이미지가 저장될 경로
let imageFilePath: String = imageDirectory.path + imageFileName
// 이미지 저장에 실패할 경우 실패 응답 보내기
do {
try imageFile.copyTo(path: imageFilePath, overWrite: false)
} catch {
sendCompleteResponse(response, status: .internalServerError, jsonBody: [JSONKey.error:error.localizedDescription])
return
}
이미지 복사도 마쳤으니 이제 데이터베이스에 정보를 저장할 차례입니다. mongoDB의 컬렉션도 불러오고, DB에 저장할 데이터를 담은 딕셔너리를 생성해줍니다.
// mongoDB 컬렉션 가져오기
guard let collection = DB.collection else {
sendCompleteResponse(response, status: .internalServerError, jsonBody: [JSONKey.error:"data base initialize failed"])
return
}
// DB에 저장할 데이터
// 차후에 응답 데이터로도 사용합니다
var jsonDictionary = [JSONKey.userName: userName,
JSONKey.description: request.param(name: JSONKey.description),
JSONKey.title: title,
JSONKey.imagePath:imageFilePath]
이제 실질적으로 DB에 저장합니다. 저장을 하면서 문서의 고유ID도 생성해줍니다. 만약 DB 저장에 실패한다면 아까 복사해온 이미지 파일은 서버 디렉터리에서 삭제해주어야 합니다.
// DB에 저장
do {
let jsonString = try jsonDictionary.jsonEncodedString()
let document: BSON = try BSON(json: jsonString)
// 문서 사용 후에는 닫아주는게 좋겠습니다
defer {
document.close()
}
// 문서 고유 아이디 생성 및 아이디 부여
let oid: BSON.OID = BSON.OID(imageFileName)
document.append(oid: oid)
// 컬렉션에 문서 저장
let result = collection.save(document: document)
// 고유 아이디를 응답 데이터에 추가
jsonDictionary[JSONKey.articleId] = oid.description
if case .success = result { } else {
sendCompleteResponse(response, status: .internalServerError, jsonBody: [JSONKey.error:result])
return
}
} catch {
// DB 저장에 실패했으므로 이미지 파일은 삭제
File(imageFilePath).delete()
Log.error(message: error.localizedDescription)
sendCompleteResponse(response, status: .internalServerError, jsonBody: [JSONKey.error:error.localizedDescription])
return
}
DB 저장을 완료하면 응답 데이터를 보내 게시물 등록 처리를 완료합니다.
// 응답 데이터에는 이미지 경로 대신 이미지 URL 전송
jsonDictionary[JSONKey.imagePath] = nil
jsonDictionary[JSONKey.imageUrl] = server.serverAddress + ":\(server.serverPort)" + "/image/" + imageFileName
// 모든 작업을 완료하고 JSON 응답
sendCompleteResponse(response, status: .created, jsonBody: jsonDictionary)
자 이렇게 postArticleHandler(request:response:)
함수 작성을 마쳤습니다.
// 사진 게시물 등록
func postArticleHandler(request: HTTPRequest, response: HTTPResponse) {
// 응답 컨텐츠 형식은 JSON
response.setHeader(.contentType, value: ContentsType.json)
// 부족한 매개변수가 없는지 확인
if let lakedParams = lakedParams(paramsNeeded: [JSONKey.userName, JSONKey.title], paramsReceived: request.postParams) {
sendCompleteResponse(response, status: .badRequest, jsonBody: [JSONKey.message:"need more params \(lakedParams)"])
}
// 이미지 파일이 업로드 되었는지 확인
guard let imageInformation: MimeReader.BodySpec = request.postFileUploads?.first,
let imageFile = imageInformation.file else {
sendCompleteResponse(response, status: .badRequest, jsonBody: [JSONKey.message:"need image file"])
return
}
// 사용자 이름, 제목 추출
guard let userName = request.param(name: JSONKey.userName),
let title = request.param(name: JSONKey.title) else {
sendCompleteResponse(response, status: .internalServerError, jsonBody: nil)
return
}
// 이미지가 저장될 디렉터리
guard let imageDirectory = imageDir else {
sendCompleteResponse(response, status: .internalServerError, jsonBody: nil)
return
}
// 고유한 이미지 이름을 위해 타임스템프 값을 활용
let timestamp: Int = icuDateToSeconds(getNow())
// 사용자이름_타임스템프.jpg 형식으로 파일이름 지정
let imageFileName: String = userName + "_" + String(timestamp) + ".jpg"
// 이미지가 저장될 경로
let imageFilePath: String = imageDirectory.path + imageFileName
// 이미지 저장에 실패할 경우 실패 응답 보내기
do {
try imageFile.copyTo(path: imageFilePath, overWrite: false)
} catch {
sendCompleteResponse(response, status: .internalServerError, jsonBody: [JSONKey.error:error.localizedDescription])
return
}
// mongoDB 컬렉션 가져오기
guard let collection = DB.collection else {
sendCompleteResponse(response, status: .internalServerError, jsonBody: [JSONKey.error:"data base initialize failed"])
return
}
// DB에 저장할 데이터
// 차후에 응답 데이터로도 사용합니다
var jsonDictionary = [JSONKey.userName: userName,
JSONKey.description: request.param(name: JSONKey.description),
JSONKey.title: title,
JSONKey.imagePath:imageFilePath]
// DB에 저장
do {
let jsonString = try jsonDictionary.jsonEncodedString()
let document: BSON = try BSON(json: jsonString)
// 문서 고유 아이디 생성 및 아이디 부여
let oid: BSON.OID = BSON.OID(imageFileName)
document.append(oid: oid)
// 컬렉션에 문서 저장
let result = collection.save(document: document)
// 고유 아이디를 응답 데이터에 추가
jsonDictionary[JSONKey.articleId] = oid.description
if case .success = result { } else {
sendCompleteResponse(response, status: .internalServerError, jsonBody: [JSONKey.error:result])
return
}
} catch {
// DB 저장에 실패했으므로 이미지 파일은 삭제
File(imageFilePath).delete()
Log.error(message: error.localizedDescription)
sendCompleteResponse(response, status: .internalServerError, jsonBody: [JSONKey.error:error.localizedDescription])
return
}
// 응답 데이터에는 이미지 경로 대신 이미지 URL 전송
jsonDictionary[JSONKey.imagePath] = nil
jsonDictionary[JSONKey.imageUrl] = server.serverAddress + ":\(server.serverPort)" + "/image/" + imageFileName
// 모든 작업을 완료하고 JSON 응답
sendCompleteResponse(response, status: .created, jsonBody: jsonDictionary)
}
자 이제 mongoDB도 실행된 상태에서, Perfect 서버 애플리케이션을 실행해봅니다. 그리고는 /article 경로에 새로운 게시물 등록을 요청해봅니다.
짜잔!! 성공했습니다! 올바른 응답이 오고 있어요!
URL을 통해 사진 GET
자, 위 이미지의 응답 데이터를 보면 image_url이 오고 있습니다. 그 URL을 통해서 이미지를 한 번 받아와볼까요?
아니, 응답받은 URL인데...!! 왜 다운로드가 안되는거죠!? 흐음... 라우팅을 뭔가 해줘야 할 것 같습니다.
으음... 자 그럼 요청을 핸들링 할 수 있는 함수를 하나 또 만들어줍니다. handlers.swift 파일에 아래 함수를 하나 만들어줬습니다. 요청의 주소의 마지막에 위치한 이미지파일의 이름을 가지고 이미지 디렉터리의 파일이 있는지 확인하여 파일 핸들러를 사용하여 응답합니다.
// 이미지 파일
func imageHandler(request: HTTPRequest, response: HTTPResponse) {
response.setHeader(.contentType, value: ContentsType.formData)
guard let imageDirectory = imageDir else {
sendCompleteResponse(response, status: .internalServerError)
return
}
guard let imageFileName = request.pathComponents.last, imageFileName.contains(".jpg") else {
sendCompleteResponse(response, status: .badRequest)
return
}
request.path = imageFileName
let handler = StaticFileHandler(documentRoot: imageDirectory.path)
handler.handleRequest(request: request, response: response)
}
main.swift 파일로 이동하여 길을 하나 더 뚫어줍니다. 일단 articleURI
상수 아래에 imageURI
도 하나 선언해줍니다.
let articleURI = "/article"
let imageURI = "/image"
let subArticleURI = articleURI + "/*"
그리고 서버 객체를 생성하는 코드 윗 줄에 아래 코드를 작성해줍니다.
routes.add(method: .get, uri: imageURI + "/*", handler: imageHandler(request:response:))
이렇게 작성하면 /image/xxx.jpg 형식으로 요청이 들어오면 imageHandler(request:response:)
함수가 핸들링 하게 될 것입니다.
자 이제 서버를 재시작하고, 다시 요청해볼까요?
으헿, 잘 나왔네요!
자 이번에는 CRUD 중, Create 먼저 해봤습니다. 다음 번에는 Read를 함께 해봐요~!
뿅!
참고문서
- PerfectDocs - Routing
- PerfectDocs - HTTPResponse
- PerfectDocs - HTTPRequest
- PerfectDocs - MongoDB
- PerfectDocs - Static File Content
by yagom
facebook : https://www.facebook.com/yagompage
facebook group : https://www.facebook.com/groups/yagom
p.s 제 포스팅을 RSS 피드로 받아보실 수 있습니다.
RSS Feed 받기