Rework book server and client.
parent
f1ad97b227
commit
2a25340ce4
@ -0,0 +1,12 @@
|
|||||||
|
set(SRCS
|
||||||
|
book.cc
|
||||||
|
book.h
|
||||||
|
book_json.cc
|
||||||
|
book_json.h
|
||||||
|
book_client.cc
|
||||||
|
book_client.h
|
||||||
|
main.cc
|
||||||
|
)
|
||||||
|
|
||||||
|
add_executable(book_client ${SRCS})
|
||||||
|
target_link_libraries(book_client ${EXAMPLE_LIBS} jsoncpp)
|
@ -0,0 +1,11 @@
|
|||||||
|
#include "book.h"
|
||||||
|
|
||||||
|
#include <iostream>
|
||||||
|
|
||||||
|
const Book kNullBook{};
|
||||||
|
|
||||||
|
std::ostream& operator<<(std::ostream& os, const Book& book) {
|
||||||
|
os << "{ " << book.id << ", " << book.title << ", " << book.price << ", "
|
||||||
|
<< book.photo << " }";
|
||||||
|
return os;
|
||||||
|
}
|
@ -0,0 +1,22 @@
|
|||||||
|
#ifndef BOOK_H_
|
||||||
|
#define BOOK_H_
|
||||||
|
|
||||||
|
#include <iosfwd>
|
||||||
|
#include <string>
|
||||||
|
|
||||||
|
struct Book {
|
||||||
|
std::string id;
|
||||||
|
std::string title;
|
||||||
|
double price;
|
||||||
|
std::string photo; // Name only
|
||||||
|
|
||||||
|
bool IsNull() const {
|
||||||
|
return id.empty();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
std::ostream& operator<<(std::ostream& os, const Book& book);
|
||||||
|
|
||||||
|
extern const Book kNullBook;
|
||||||
|
|
||||||
|
#endif // BOOK_H_
|
@ -0,0 +1,194 @@
|
|||||||
|
#include "book_client.h"
|
||||||
|
|
||||||
|
#include <iostream>
|
||||||
|
|
||||||
|
#include "boost/algorithm/string/predicate.hpp"
|
||||||
|
#include "boost/filesystem/operations.hpp"
|
||||||
|
#include "json/json.h"
|
||||||
|
|
||||||
|
#include "book_json.h"
|
||||||
|
|
||||||
|
BookClient::BookClient(const std::string& url, int timeout)
|
||||||
|
: url_(url), session_(timeout) {
|
||||||
|
// Default Content-Type for requests who have a body.
|
||||||
|
session_.set_media_type("application/json");
|
||||||
|
session_.set_charset("utf-8");
|
||||||
|
}
|
||||||
|
|
||||||
|
bool BookClient::Query(std::list<Book>* books) {
|
||||||
|
try {
|
||||||
|
auto r = session_.Send(WEBCC_GET(url_).Path("books")());
|
||||||
|
|
||||||
|
if (!CheckStatus(r, webcc::Status::kOK)) {
|
||||||
|
// Response HTTP status error.
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
Json::Value json = StringToJson(r->data());
|
||||||
|
|
||||||
|
if (!json.isArray()) {
|
||||||
|
return false; // Should be a JSON array of books.
|
||||||
|
}
|
||||||
|
|
||||||
|
for (Json::ArrayIndex i = 0; i < json.size(); ++i) {
|
||||||
|
books->push_back(JsonToBook(json[i]));
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
|
||||||
|
} catch (const webcc::Error& error) {
|
||||||
|
std::cerr << error << std::endl;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
bool BookClient::Create(const std::string& title, double price,
|
||||||
|
std::string* id) {
|
||||||
|
Json::Value req_json;
|
||||||
|
req_json["title"] = title;
|
||||||
|
req_json["price"] = price;
|
||||||
|
|
||||||
|
try {
|
||||||
|
auto r = session_.Send(WEBCC_POST(url_).Path("books").
|
||||||
|
Body(JsonToString(req_json))());
|
||||||
|
|
||||||
|
if (!CheckStatus(r, webcc::Status::kCreated)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
Json::Value rsp_json = StringToJson(r->data());
|
||||||
|
*id = rsp_json["id"].asString();
|
||||||
|
|
||||||
|
if (id->empty()) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
|
||||||
|
} catch (const webcc::Error& error) {
|
||||||
|
std::cerr << error << std::endl;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
bool BookClient::Get(const std::string& id, Book* book) {
|
||||||
|
try {
|
||||||
|
auto r = session_.Send(WEBCC_GET(url_).Path("books").Path(id)());
|
||||||
|
|
||||||
|
if (!CheckStatus(r, webcc::Status::kOK)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return JsonStringToBook(r->data(), book);
|
||||||
|
|
||||||
|
} catch (const webcc::Error& error) {
|
||||||
|
std::cerr << error << std::endl;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
bool BookClient::Set(const std::string& id, const std::string& title,
|
||||||
|
double price) {
|
||||||
|
Json::Value json;
|
||||||
|
json["title"] = title;
|
||||||
|
json["price"] = price;
|
||||||
|
|
||||||
|
try {
|
||||||
|
auto r = session_.Send(WEBCC_PUT(url_).Path("books").Path(id).
|
||||||
|
Body(JsonToString(json))());
|
||||||
|
|
||||||
|
if (!CheckStatus(r, webcc::Status::kOK)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
|
||||||
|
} catch (const webcc::Error& error) {
|
||||||
|
std::cerr << error << std::endl;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
bool BookClient::Delete(const std::string& id) {
|
||||||
|
try {
|
||||||
|
auto r = session_.Send(WEBCC_DELETE(url_).Path("books").Path(id)());
|
||||||
|
|
||||||
|
if (!CheckStatus(r, webcc::Status::kOK)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
|
||||||
|
} catch (const webcc::Error& error) {
|
||||||
|
std::cerr << error << std::endl;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
bool BookClient::GetPhoto(const std::string& id, const bfs::path& path) {
|
||||||
|
try {
|
||||||
|
auto r = session_.Send(WEBCC_GET(url_).
|
||||||
|
Path("books").Path(id).Path("photo")(),
|
||||||
|
true); // Save to temp file
|
||||||
|
|
||||||
|
if (!CheckStatus(r, webcc::Status::kOK)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
r->file_body()->Move(path);
|
||||||
|
|
||||||
|
return true;
|
||||||
|
|
||||||
|
} catch (const webcc::Error& error) {
|
||||||
|
std::cerr << error << std::endl;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
bool BookClient::SetPhoto(const std::string& id, const bfs::path& path) {
|
||||||
|
try {
|
||||||
|
if (!CheckPhoto(path)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto r = session_.Send(WEBCC_PUT(url_).
|
||||||
|
Path("books").Path(id).Path("photo").
|
||||||
|
File(path)());
|
||||||
|
|
||||||
|
if (!CheckStatus(r, webcc::Status::kOK)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
|
||||||
|
} catch (const webcc::Error& error) {
|
||||||
|
std::cerr << error << std::endl;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
bool BookClient::CheckPhoto(const bfs::path& photo) {
|
||||||
|
if (photo.empty()) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!bfs::is_regular_file(photo) || !bfs::exists(photo)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto ext = photo.extension().string();
|
||||||
|
if (!boost::iequals(ext, ".jpg") && !boost::iequals(ext, ".jpeg")) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool BookClient::CheckStatus(webcc::ResponsePtr response, int expected_status) {
|
||||||
|
if (response->status() != expected_status) {
|
||||||
|
std::cerr << "HTTP status error (actual: " << response->status()
|
||||||
|
<< "expected: " << expected_status << ")." << std::endl;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
@ -0,0 +1,49 @@
|
|||||||
|
#ifndef BOOK_CLIENT_H_
|
||||||
|
#define BOOK_CLIENT_H_
|
||||||
|
|
||||||
|
#include <list>
|
||||||
|
#include <string>
|
||||||
|
|
||||||
|
#include "boost/filesystem/path.hpp"
|
||||||
|
#include "json/json-forwards.h"
|
||||||
|
|
||||||
|
#include "webcc/client_session.h"
|
||||||
|
|
||||||
|
#include "book.h"
|
||||||
|
|
||||||
|
namespace bfs = boost::filesystem;
|
||||||
|
|
||||||
|
class BookClient {
|
||||||
|
public:
|
||||||
|
explicit BookClient(const std::string& url, int timeout = 0);
|
||||||
|
|
||||||
|
~BookClient() = default;
|
||||||
|
|
||||||
|
bool Query(std::list<Book>* books);
|
||||||
|
|
||||||
|
bool Create(const std::string& title, double price, std::string* id);
|
||||||
|
|
||||||
|
bool Get(const std::string& id, Book* book);
|
||||||
|
|
||||||
|
bool Set(const std::string& id, const std::string& title, double price);
|
||||||
|
|
||||||
|
bool Delete(const std::string& id);
|
||||||
|
|
||||||
|
// Get photo, save to the given path.
|
||||||
|
bool GetPhoto(const std::string& id, const bfs::path& path);
|
||||||
|
|
||||||
|
// Set photo using the file of the given path.
|
||||||
|
bool SetPhoto(const std::string& id, const bfs::path& path);
|
||||||
|
|
||||||
|
private:
|
||||||
|
bool CheckPhoto(const bfs::path& photo);
|
||||||
|
|
||||||
|
// Check HTTP response status.
|
||||||
|
bool CheckStatus(webcc::ResponsePtr response, int expected_status);
|
||||||
|
|
||||||
|
private:
|
||||||
|
std::string url_;
|
||||||
|
webcc::ClientSession session_;
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif // BOOK_CLIENT_H_
|
@ -0,0 +1,59 @@
|
|||||||
|
#include "book_json.h"
|
||||||
|
|
||||||
|
#include <iostream>
|
||||||
|
#include <sstream>
|
||||||
|
|
||||||
|
#include "json/json.h"
|
||||||
|
|
||||||
|
#include "book.h"
|
||||||
|
|
||||||
|
std::string JsonToString(const Json::Value& json) {
|
||||||
|
Json::StreamWriterBuilder builder;
|
||||||
|
return Json::writeString(builder, json);
|
||||||
|
}
|
||||||
|
|
||||||
|
Json::Value StringToJson(const std::string& str) {
|
||||||
|
Json::Value json;
|
||||||
|
|
||||||
|
Json::CharReaderBuilder builder;
|
||||||
|
std::stringstream stream(str);
|
||||||
|
std::string errs;
|
||||||
|
if (!Json::parseFromStream(builder, stream, &json, &errs)) {
|
||||||
|
std::cerr << errs << std::endl;
|
||||||
|
}
|
||||||
|
|
||||||
|
return json;
|
||||||
|
}
|
||||||
|
|
||||||
|
Json::Value BookToJson(const Book& book) {
|
||||||
|
Json::Value json;
|
||||||
|
json["id"] = book.id;
|
||||||
|
json["title"] = book.title;
|
||||||
|
json["price"] = book.price;
|
||||||
|
json["photo"] = book.photo;
|
||||||
|
return json;
|
||||||
|
}
|
||||||
|
|
||||||
|
Book JsonToBook(const Json::Value& json) {
|
||||||
|
return {
|
||||||
|
json["id"].asString(),
|
||||||
|
json["title"].asString(),
|
||||||
|
json["price"].asDouble(),
|
||||||
|
json["photo"].asString(),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
std::string BookToJsonString(const Book& book) {
|
||||||
|
return JsonToString(BookToJson(book));
|
||||||
|
}
|
||||||
|
|
||||||
|
bool JsonStringToBook(const std::string& json_str, Book* book) {
|
||||||
|
Json::Value json = StringToJson(json_str);
|
||||||
|
|
||||||
|
if (!json) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
*book = JsonToBook(json);
|
||||||
|
return true;
|
||||||
|
}
|
@ -0,0 +1,22 @@
|
|||||||
|
#ifndef BOOK_JSON_H_
|
||||||
|
#define BOOK_JSON_H_
|
||||||
|
|
||||||
|
#include <string>
|
||||||
|
|
||||||
|
#include "json/json-forwards.h"
|
||||||
|
|
||||||
|
struct Book;
|
||||||
|
|
||||||
|
std::string JsonToString(const Json::Value& json);
|
||||||
|
|
||||||
|
Json::Value StringToJson(const std::string& str);
|
||||||
|
|
||||||
|
Json::Value BookToJson(const Book& book);
|
||||||
|
|
||||||
|
Book JsonToBook(const Json::Value& json);
|
||||||
|
|
||||||
|
std::string BookToJsonString(const Book& book);
|
||||||
|
|
||||||
|
bool JsonStringToBook(const std::string& json_str, Book* book);
|
||||||
|
|
||||||
|
#endif // BOOK_JSON_H_
|
@ -0,0 +1,141 @@
|
|||||||
|
#include <iostream>
|
||||||
|
|
||||||
|
#include "boost/filesystem/operations.hpp"
|
||||||
|
#include "webcc/logger.h"
|
||||||
|
|
||||||
|
#include "book_client.h"
|
||||||
|
|
||||||
|
// Memory leak detection with VLD.
|
||||||
|
#if (defined(_WIN32) || defined(_WIN64))
|
||||||
|
#if defined(_DEBUG) && defined(WEBCC_ENABLE_VLD)
|
||||||
|
#pragma message ("< include vld.h >")
|
||||||
|
#include "vld/vld.h"
|
||||||
|
#pragma comment(lib, "vld")
|
||||||
|
#endif
|
||||||
|
#endif
|
||||||
|
|
||||||
|
// -----------------------------------------------------------------------------
|
||||||
|
|
||||||
|
void PrintSeparator() {
|
||||||
|
static const std::string s_line(80, '-');
|
||||||
|
std::cout << s_line << std::endl;
|
||||||
|
}
|
||||||
|
|
||||||
|
void PrintBook(const Book& book) {
|
||||||
|
std::cout << "Book: " << book << std::endl;
|
||||||
|
}
|
||||||
|
|
||||||
|
void PrintBookList(const std::list<Book>& books) {
|
||||||
|
std::cout << "Book list: " << books.size() << std::endl;
|
||||||
|
for (const Book& book : books) {
|
||||||
|
std::cout << " Book: " << book << std::endl;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// -----------------------------------------------------------------------------
|
||||||
|
|
||||||
|
int main(int argc, char* argv[]) {
|
||||||
|
if (argc < 3) {
|
||||||
|
std::cout << "usage: book_client <url> <photo_dir>" << std::endl;
|
||||||
|
std::cout << "e.g.," << std::endl;
|
||||||
|
std::cout << " $ book_client http://localhost:8080 path/to/photo_dir"
|
||||||
|
<< std::endl;
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::string url = argv[1];
|
||||||
|
|
||||||
|
bfs::path photo_dir = argv[2];
|
||||||
|
if (!bfs::is_directory(photo_dir) || !bfs::exists(photo_dir)) {
|
||||||
|
std::cerr << "Invalid photo dir!" << std::endl;
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::cout << "Test photo dir: " << photo_dir << std::endl;
|
||||||
|
|
||||||
|
WEBCC_LOG_INIT("", webcc::LOG_CONSOLE_FILE_OVERWRITE);
|
||||||
|
|
||||||
|
BookClient client(url);
|
||||||
|
|
||||||
|
PrintSeparator();
|
||||||
|
|
||||||
|
std::list<Book> books;
|
||||||
|
if (client.Query(&books)) {
|
||||||
|
PrintBookList(books);
|
||||||
|
} else {
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
PrintSeparator();
|
||||||
|
|
||||||
|
std::string id;
|
||||||
|
if (client.Create("1984", 12.3, &id)) {
|
||||||
|
std::cout << "Book ID: " << id << std::endl;
|
||||||
|
} else {
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!client.SetPhoto(id, photo_dir / "1984.jpg")) {
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
PrintSeparator();
|
||||||
|
|
||||||
|
books.clear();
|
||||||
|
if (client.Query(&books)) {
|
||||||
|
PrintBookList(books);
|
||||||
|
} else {
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
PrintSeparator();
|
||||||
|
|
||||||
|
Book book;
|
||||||
|
if (client.Get(id, &book)) {
|
||||||
|
PrintBook(book);
|
||||||
|
} else {
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
PrintSeparator();
|
||||||
|
|
||||||
|
std::cout << "Press any key to continue...";
|
||||||
|
std::getchar();
|
||||||
|
|
||||||
|
if (!client.Set(id, "1Q84", 32.1)) {
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!client.SetPhoto(id, photo_dir / "1Q84.jpg")) {
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
PrintSeparator();
|
||||||
|
|
||||||
|
if (client.Get(id, &book)) {
|
||||||
|
PrintBook(book);
|
||||||
|
} else {
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
PrintSeparator();
|
||||||
|
|
||||||
|
std::cout << "Press any key to continue...";
|
||||||
|
std::getchar();
|
||||||
|
|
||||||
|
if (!client.Delete(id)) {
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
PrintSeparator();
|
||||||
|
|
||||||
|
books.clear();
|
||||||
|
if (client.Query(&books)) {
|
||||||
|
PrintBookList(books);
|
||||||
|
}
|
||||||
|
|
||||||
|
std::cout << "Press any key to continue...";
|
||||||
|
std::getchar();
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
Binary file not shown.
After Width: | Height: | Size: 59 KiB |
Binary file not shown.
After Width: | Height: | Size: 13 KiB |
@ -0,0 +1,14 @@
|
|||||||
|
set(SRCS
|
||||||
|
book.cc
|
||||||
|
book.h
|
||||||
|
book_db.cc
|
||||||
|
book_db.h
|
||||||
|
book_json.cc
|
||||||
|
book_json.h
|
||||||
|
views.cc
|
||||||
|
views.h
|
||||||
|
main.cc
|
||||||
|
)
|
||||||
|
|
||||||
|
add_executable(book_server ${SRCS})
|
||||||
|
target_link_libraries(book_server ${EXAMPLE_LIBS} jsoncpp)
|
@ -0,0 +1,11 @@
|
|||||||
|
#include "book.h"
|
||||||
|
|
||||||
|
#include <iostream>
|
||||||
|
|
||||||
|
const Book kNullBook{};
|
||||||
|
|
||||||
|
std::ostream& operator<<(std::ostream& os, const Book& book) {
|
||||||
|
os << "{ " << book.id << ", " << book.title << ", " << book.price << ", "
|
||||||
|
<< book.photo << " }";
|
||||||
|
return os;
|
||||||
|
}
|
@ -0,0 +1,24 @@
|
|||||||
|
#ifndef BOOK_H_
|
||||||
|
#define BOOK_H_
|
||||||
|
|
||||||
|
#include <iosfwd>
|
||||||
|
#include <string>
|
||||||
|
|
||||||
|
#include "boost/filesystem/path.hpp"
|
||||||
|
|
||||||
|
struct Book {
|
||||||
|
std::string id;
|
||||||
|
std::string title;
|
||||||
|
double price;
|
||||||
|
std::string photo; // Name only
|
||||||
|
|
||||||
|
bool IsNull() const {
|
||||||
|
return id.empty();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
std::ostream& operator<<(std::ostream& os, const Book& book);
|
||||||
|
|
||||||
|
extern const Book kNullBook;
|
||||||
|
|
||||||
|
#endif // BOOK_H_
|
@ -0,0 +1,66 @@
|
|||||||
|
#include "book_db.h"
|
||||||
|
|
||||||
|
#include <algorithm>
|
||||||
|
|
||||||
|
const Book& BookDB::Get(const std::string& id) const {
|
||||||
|
auto it = Find(id);
|
||||||
|
return (it == books_.end() ? kNullBook : *it);
|
||||||
|
}
|
||||||
|
|
||||||
|
std::string BookDB::Add(const Book& book) {
|
||||||
|
std::string id = NewID();
|
||||||
|
books_.push_back({ id, book.title, book.price });
|
||||||
|
return id;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool BookDB::Set(const Book& book) {
|
||||||
|
auto it = Find(book.id);
|
||||||
|
if (it != books_.end()) {
|
||||||
|
it->title = book.title;
|
||||||
|
it->price = book.price;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::string BookDB::GetPhoto(const std::string& id) const {
|
||||||
|
auto it = Find(id);
|
||||||
|
return it != books_.end() ? it->photo : "";
|
||||||
|
}
|
||||||
|
|
||||||
|
bool BookDB::SetPhoto(const std::string& id, const std::string& photo) {
|
||||||
|
auto it = Find(id);
|
||||||
|
if (it != books_.end()) {
|
||||||
|
it->photo = photo;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool BookDB::Delete(const std::string& id) {
|
||||||
|
auto it = Find(id);
|
||||||
|
|
||||||
|
if (it != books_.end()) {
|
||||||
|
books_.erase(it);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::list<Book>::const_iterator BookDB::Find(const std::string& id) const {
|
||||||
|
return std::find_if(books_.begin(), books_.end(),
|
||||||
|
[&id](const Book& book) { return book.id == id; });
|
||||||
|
}
|
||||||
|
|
||||||
|
std::list<Book>::iterator BookDB::Find(const std::string& id) {
|
||||||
|
return std::find_if(books_.begin(), books_.end(),
|
||||||
|
[&id](Book& book) { return book.id == id; });
|
||||||
|
}
|
||||||
|
|
||||||
|
std::string BookDB::NewID() const {
|
||||||
|
static int s_id_counter = 0;
|
||||||
|
|
||||||
|
++s_id_counter;
|
||||||
|
return std::to_string(s_id_counter);
|
||||||
|
}
|
@ -0,0 +1,44 @@
|
|||||||
|
#ifndef BOOK_DB_H_
|
||||||
|
#define BOOK_DB_H_
|
||||||
|
|
||||||
|
#include <list>
|
||||||
|
#include <string>
|
||||||
|
|
||||||
|
#include "book.h"
|
||||||
|
|
||||||
|
// Book database simulator.
|
||||||
|
// There should be some database in a real product.
|
||||||
|
|
||||||
|
class BookDB {
|
||||||
|
public:
|
||||||
|
const std::list<Book>& books() const {
|
||||||
|
return books_;
|
||||||
|
}
|
||||||
|
|
||||||
|
const Book& Get(const std::string& id) const;
|
||||||
|
|
||||||
|
// Add a book, return the ID.
|
||||||
|
// NOTE: The ID of the input book will be ignored so should be empty.
|
||||||
|
std::string Add(const Book& book);
|
||||||
|
|
||||||
|
bool Set(const Book& book);
|
||||||
|
|
||||||
|
std::string GetPhoto(const std::string& id) const;
|
||||||
|
|
||||||
|
bool SetPhoto(const std::string& id, const std::string& photo);
|
||||||
|
|
||||||
|
bool Delete(const std::string& id);
|
||||||
|
|
||||||
|
private:
|
||||||
|
std::list<Book>::const_iterator Find(const std::string& id) const;
|
||||||
|
|
||||||
|
std::list<Book>::iterator Find(const std::string& id);
|
||||||
|
|
||||||
|
// Allocate a new book ID.
|
||||||
|
std::string NewID() const;
|
||||||
|
|
||||||
|
private:
|
||||||
|
std::list<Book> books_;
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif // BOOK_DB_H_
|
@ -0,0 +1,57 @@
|
|||||||
|
#include "book_json.h"
|
||||||
|
|
||||||
|
#include <iostream>
|
||||||
|
#include <sstream>
|
||||||
|
|
||||||
|
#include "json/json.h"
|
||||||
|
|
||||||
|
#include "book.h"
|
||||||
|
|
||||||
|
std::string JsonToString(const Json::Value& json) {
|
||||||
|
Json::StreamWriterBuilder builder;
|
||||||
|
return Json::writeString(builder, json);
|
||||||
|
}
|
||||||
|
|
||||||
|
Json::Value StringToJson(const std::string& str) {
|
||||||
|
Json::Value json;
|
||||||
|
|
||||||
|
Json::CharReaderBuilder builder;
|
||||||
|
std::stringstream stream(str);
|
||||||
|
std::string errs;
|
||||||
|
if (!Json::parseFromStream(builder, stream, &json, &errs)) {
|
||||||
|
std::cerr << errs << std::endl;
|
||||||
|
}
|
||||||
|
|
||||||
|
return json;
|
||||||
|
}
|
||||||
|
|
||||||
|
Json::Value BookToJson(const Book& book) {
|
||||||
|
Json::Value json;
|
||||||
|
json["id"] = book.id;
|
||||||
|
json["title"] = book.title;
|
||||||
|
json["price"] = book.price;
|
||||||
|
return json;
|
||||||
|
}
|
||||||
|
|
||||||
|
Book JsonToBook(const Json::Value& json) {
|
||||||
|
return {
|
||||||
|
json["id"].asString(),
|
||||||
|
json["title"].asString(),
|
||||||
|
json["price"].asDouble(),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
std::string BookToJsonString(const Book& book) {
|
||||||
|
return JsonToString(BookToJson(book));
|
||||||
|
}
|
||||||
|
|
||||||
|
bool JsonStringToBook(const std::string& json_str, Book* book) {
|
||||||
|
Json::Value json = StringToJson(json_str);
|
||||||
|
|
||||||
|
if (!json) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
*book = JsonToBook(json);
|
||||||
|
return true;
|
||||||
|
}
|
@ -0,0 +1,22 @@
|
|||||||
|
#ifndef BOOK_JSON_H_
|
||||||
|
#define BOOK_JSON_H_
|
||||||
|
|
||||||
|
#include <string>
|
||||||
|
|
||||||
|
#include "json/json-forwards.h"
|
||||||
|
|
||||||
|
struct Book;
|
||||||
|
|
||||||
|
std::string JsonToString(const Json::Value& json);
|
||||||
|
|
||||||
|
Json::Value StringToJson(const std::string& str);
|
||||||
|
|
||||||
|
Json::Value BookToJson(const Book& book);
|
||||||
|
|
||||||
|
Book JsonToBook(const Json::Value& json);
|
||||||
|
|
||||||
|
std::string BookToJsonString(const Book& book);
|
||||||
|
|
||||||
|
bool JsonStringToBook(const std::string& json_str, Book* book);
|
||||||
|
|
||||||
|
#endif // BOOK_JSON_H_
|
@ -0,0 +1,68 @@
|
|||||||
|
#include <iostream>
|
||||||
|
|
||||||
|
#include "boost/filesystem/operations.hpp"
|
||||||
|
|
||||||
|
#include "webcc/logger.h"
|
||||||
|
#include "webcc/server.h"
|
||||||
|
|
||||||
|
#include "views.h"
|
||||||
|
|
||||||
|
// Memory leak detection with VLD.
|
||||||
|
#if (defined(_WIN32) || defined(_WIN64))
|
||||||
|
#if defined(_DEBUG) && defined(WEBCC_ENABLE_VLD)
|
||||||
|
#pragma message ("< include vld.h >")
|
||||||
|
#include "vld/vld.h"
|
||||||
|
#pragma comment(lib, "vld")
|
||||||
|
#endif
|
||||||
|
#endif
|
||||||
|
|
||||||
|
int main(int argc, char* argv[]) {
|
||||||
|
if (argc < 3) {
|
||||||
|
std::cout << "usage: book_server <port> <upload_dir>" << std::endl;
|
||||||
|
std::cout << "e.g.," << std::endl;
|
||||||
|
std::cout << " $ book_server 8080 D:/upload" << std::endl;
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
WEBCC_LOG_INIT("", webcc::LOG_CONSOLE);
|
||||||
|
|
||||||
|
std::uint16_t port = static_cast<std::uint16_t>(std::atoi(argv[1]));
|
||||||
|
|
||||||
|
bfs::path upload_dir = argv[2];
|
||||||
|
if (!bfs::is_directory(upload_dir) || !bfs::exists(upload_dir)) {
|
||||||
|
std::cerr << "Invalid upload dir!" << std::endl;
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Add a sub-dir for book photos.
|
||||||
|
bfs::path photo_dir = upload_dir / "books";
|
||||||
|
if (!bfs::exists(photo_dir)) {
|
||||||
|
bfs::create_directory(photo_dir);
|
||||||
|
}
|
||||||
|
|
||||||
|
std::cout << "Book photos will be saved to: " << photo_dir << std::endl;
|
||||||
|
|
||||||
|
try {
|
||||||
|
webcc::Server server(port); // No doc root
|
||||||
|
|
||||||
|
server.Route("/books",
|
||||||
|
std::make_shared<BookListView>(),
|
||||||
|
{ "GET", "POST" });
|
||||||
|
|
||||||
|
server.Route(webcc::R("/books/(\\d+)"),
|
||||||
|
std::make_shared<BookDetailView>(photo_dir),
|
||||||
|
{ "GET", "PUT", "DELETE" });
|
||||||
|
|
||||||
|
server.Route(webcc::R("/books/(\\d+)/photo"),
|
||||||
|
std::make_shared<BookPhotoView>(photo_dir),
|
||||||
|
{ "GET", "PUT", "DELETE" });
|
||||||
|
|
||||||
|
server.Run(2);
|
||||||
|
|
||||||
|
} catch (const std::exception& e) {
|
||||||
|
std::cerr << e.what() << std::endl;
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
@ -0,0 +1,223 @@
|
|||||||
|
#include "views.h"
|
||||||
|
|
||||||
|
#include "json/json.h"
|
||||||
|
|
||||||
|
#include "boost/filesystem/operations.hpp"
|
||||||
|
#include "webcc/response_builder.h"
|
||||||
|
|
||||||
|
#include "book.h"
|
||||||
|
#include "book_db.h"
|
||||||
|
#include "book_json.h"
|
||||||
|
|
||||||
|
// -----------------------------------------------------------------------------
|
||||||
|
|
||||||
|
static BookDB g_book_db;
|
||||||
|
|
||||||
|
// -----------------------------------------------------------------------------
|
||||||
|
|
||||||
|
webcc::ResponsePtr BookListView::Handle(webcc::RequestPtr request) {
|
||||||
|
if (request->method() == "GET") {
|
||||||
|
return Get(request);
|
||||||
|
}
|
||||||
|
if (request->method() == "POST") {
|
||||||
|
return Post(request);
|
||||||
|
}
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
|
||||||
|
webcc::ResponsePtr BookListView::Get(webcc::RequestPtr request) {
|
||||||
|
Json::Value json(Json::arrayValue);
|
||||||
|
|
||||||
|
for (const Book& book : g_book_db.books()) {
|
||||||
|
json.append(BookToJson(book));
|
||||||
|
}
|
||||||
|
|
||||||
|
// Return all books as a JSON array.
|
||||||
|
|
||||||
|
return webcc::ResponseBuilder{}.OK().Body(JsonToString(json)).Json().Utf8()();
|
||||||
|
}
|
||||||
|
|
||||||
|
webcc::ResponsePtr BookListView::Post(webcc::RequestPtr request) {
|
||||||
|
Book book;
|
||||||
|
if (JsonStringToBook(request->data(), &book)) {
|
||||||
|
std::string id = g_book_db.Add(book);
|
||||||
|
|
||||||
|
Json::Value json;
|
||||||
|
json["id"] = id;
|
||||||
|
|
||||||
|
return webcc::ResponseBuilder{}.Created().Body(JsonToString(json)).Json().Utf8()();
|
||||||
|
} else {
|
||||||
|
// Invalid JSON
|
||||||
|
return webcc::ResponseBuilder{}.BadRequest()();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// -----------------------------------------------------------------------------
|
||||||
|
|
||||||
|
BookDetailView::BookDetailView(bfs::path photo_dir)
|
||||||
|
: photo_dir_(std::move(photo_dir)) {
|
||||||
|
}
|
||||||
|
|
||||||
|
webcc::ResponsePtr BookDetailView::Handle(webcc::RequestPtr request) {
|
||||||
|
if (request->method() == "GET") {
|
||||||
|
return Get(request);
|
||||||
|
}
|
||||||
|
if (request->method() == "PUT") {
|
||||||
|
return Put(request);
|
||||||
|
}
|
||||||
|
if (request->method() == "DELETE") {
|
||||||
|
return Delete(request);
|
||||||
|
}
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
|
||||||
|
webcc::ResponsePtr BookDetailView::Get(webcc::RequestPtr request) {
|
||||||
|
if (request->args().size() != 1) {
|
||||||
|
// NotFound means the resource specified by the URL cannot be found.
|
||||||
|
// BadRequest could be another choice.
|
||||||
|
return webcc::ResponseBuilder{}.NotFound()();
|
||||||
|
}
|
||||||
|
|
||||||
|
const std::string& id = request->args()[0];
|
||||||
|
|
||||||
|
const Book& book = g_book_db.Get(id);
|
||||||
|
if (book.IsNull()) {
|
||||||
|
return webcc::ResponseBuilder{}.NotFound()();
|
||||||
|
}
|
||||||
|
|
||||||
|
return webcc::ResponseBuilder{}.OK().Body(BookToJsonString(book)).Json().Utf8()();
|
||||||
|
}
|
||||||
|
|
||||||
|
webcc::ResponsePtr BookDetailView::Put(webcc::RequestPtr request) {
|
||||||
|
if (request->args().size() != 1) {
|
||||||
|
return webcc::ResponseBuilder{}.NotFound()();
|
||||||
|
}
|
||||||
|
|
||||||
|
const std::string& id = request->args()[0];
|
||||||
|
|
||||||
|
Book book;
|
||||||
|
if (!JsonStringToBook(request->data(), &book)) {
|
||||||
|
return webcc::ResponseBuilder{}.BadRequest()();
|
||||||
|
}
|
||||||
|
|
||||||
|
book.id = id;
|
||||||
|
g_book_db.Set(book);
|
||||||
|
|
||||||
|
return webcc::ResponseBuilder{}.OK()();
|
||||||
|
}
|
||||||
|
|
||||||
|
webcc::ResponsePtr BookDetailView::Delete(webcc::RequestPtr request) {
|
||||||
|
if (request->args().size() != 1) {
|
||||||
|
return webcc::ResponseBuilder{}.NotFound()();
|
||||||
|
}
|
||||||
|
|
||||||
|
const std::string& id = request->args()[0];
|
||||||
|
|
||||||
|
std::string photo_name = g_book_db.GetPhoto(id);
|
||||||
|
|
||||||
|
// Delete the book from DB.
|
||||||
|
if (!g_book_db.Delete(id)) {
|
||||||
|
return webcc::ResponseBuilder{}.NotFound()();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Delete the photo from file system.
|
||||||
|
if (!photo_name.empty()) {
|
||||||
|
boost::system::error_code ec;
|
||||||
|
bfs::remove(photo_dir_ / photo_name, ec);
|
||||||
|
}
|
||||||
|
|
||||||
|
return webcc::ResponseBuilder{}.OK()();
|
||||||
|
}
|
||||||
|
|
||||||
|
// -----------------------------------------------------------------------------
|
||||||
|
|
||||||
|
BookPhotoView::BookPhotoView(bfs::path photo_dir)
|
||||||
|
: photo_dir_(std::move(photo_dir)) {
|
||||||
|
}
|
||||||
|
|
||||||
|
webcc::ResponsePtr BookPhotoView::Handle(webcc::RequestPtr request) {
|
||||||
|
if (request->method() == "GET") {
|
||||||
|
return Get(request);
|
||||||
|
}
|
||||||
|
if (request->method() == "PUT") {
|
||||||
|
return Put(request);
|
||||||
|
}
|
||||||
|
if (request->method() == "DELETE") {
|
||||||
|
return Delete(request);
|
||||||
|
}
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO: Check content type to see if it's JPEG.
|
||||||
|
webcc::ResponsePtr BookPhotoView::Get(webcc::RequestPtr request) {
|
||||||
|
if (request->args().size() != 1) {
|
||||||
|
return webcc::ResponseBuilder{}.NotFound()();
|
||||||
|
}
|
||||||
|
|
||||||
|
const std::string& id = request->args()[0];
|
||||||
|
|
||||||
|
const Book& book = g_book_db.Get(id);
|
||||||
|
if (book.IsNull() || book.photo.empty()) {
|
||||||
|
return webcc::ResponseBuilder{}.NotFound()();
|
||||||
|
}
|
||||||
|
|
||||||
|
bfs::path photo_path = photo_dir_ / book.photo;
|
||||||
|
if (!bfs::exists(photo_path)) {
|
||||||
|
return webcc::ResponseBuilder{}.NotFound()();
|
||||||
|
}
|
||||||
|
|
||||||
|
// File() might throw Error::kFileError.
|
||||||
|
// TODO: Avoid exception handling.
|
||||||
|
try {
|
||||||
|
return webcc::ResponseBuilder{}.OK().File(photo_path)();
|
||||||
|
} catch (const webcc::Error&) {
|
||||||
|
return webcc::ResponseBuilder{}.NotFound()();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
webcc::ResponsePtr BookPhotoView::Put(webcc::RequestPtr request) {
|
||||||
|
if (request->args().size() != 1) {
|
||||||
|
return webcc::ResponseBuilder{}.NotFound()();
|
||||||
|
}
|
||||||
|
|
||||||
|
const std::string& id = request->args()[0];
|
||||||
|
|
||||||
|
const Book& book = g_book_db.Get(id);
|
||||||
|
if (book.IsNull()) {
|
||||||
|
return webcc::ResponseBuilder{}.NotFound()();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Use ID as the name of the photo.
|
||||||
|
// You can also use a UUID or any unique string as the name.
|
||||||
|
auto photo_name = id + ".jpg";
|
||||||
|
|
||||||
|
if (!request->file_body()->Move(photo_dir_ / photo_name)) {
|
||||||
|
return webcc::ResponseBuilder{}.InternalServerError()();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Set photo name to DB.
|
||||||
|
if (!g_book_db.SetPhoto(id, photo_name)) {
|
||||||
|
return webcc::ResponseBuilder{}.InternalServerError()();
|
||||||
|
}
|
||||||
|
|
||||||
|
return webcc::ResponseBuilder{}.OK()();
|
||||||
|
}
|
||||||
|
|
||||||
|
webcc::ResponsePtr BookPhotoView::Delete(webcc::RequestPtr request) {
|
||||||
|
if (request->args().size() != 1) {
|
||||||
|
return webcc::ResponseBuilder{}.NotFound()();
|
||||||
|
}
|
||||||
|
|
||||||
|
const std::string& id = request->args()[0];
|
||||||
|
|
||||||
|
const Book& book = g_book_db.Get(id);
|
||||||
|
if (book.IsNull() || book.photo.empty()) {
|
||||||
|
return webcc::ResponseBuilder{}.NotFound()();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Error handling is simplified.
|
||||||
|
boost::system::error_code ec;
|
||||||
|
bfs::remove(photo_dir_ / book.photo, ec);
|
||||||
|
|
||||||
|
return webcc::ResponseBuilder{}.OK()();
|
||||||
|
}
|
@ -0,0 +1,76 @@
|
|||||||
|
#ifndef VIEWS_H_
|
||||||
|
#define VIEWS_H_
|
||||||
|
|
||||||
|
#include "webcc/view.h"
|
||||||
|
|
||||||
|
#include "boost/filesystem/path.hpp"
|
||||||
|
|
||||||
|
namespace bfs = boost::filesystem;
|
||||||
|
|
||||||
|
// -----------------------------------------------------------------------------
|
||||||
|
|
||||||
|
// URL: /books
|
||||||
|
class BookListView : public webcc::View {
|
||||||
|
public:
|
||||||
|
webcc::ResponsePtr Handle(webcc::RequestPtr request) override;
|
||||||
|
|
||||||
|
private:
|
||||||
|
// Get a list of books based on query parameters.
|
||||||
|
webcc::ResponsePtr Get(webcc::RequestPtr request);
|
||||||
|
|
||||||
|
// Create a new book.
|
||||||
|
webcc::ResponsePtr Post(webcc::RequestPtr request);
|
||||||
|
};
|
||||||
|
|
||||||
|
// -----------------------------------------------------------------------------
|
||||||
|
|
||||||
|
// URL: /books/{id}
|
||||||
|
class BookDetailView : public webcc::View {
|
||||||
|
public:
|
||||||
|
explicit BookDetailView(bfs::path photo_dir);
|
||||||
|
|
||||||
|
webcc::ResponsePtr Handle(webcc::RequestPtr request) override;
|
||||||
|
|
||||||
|
private:
|
||||||
|
// Get the detailed information of a book.
|
||||||
|
webcc::ResponsePtr Get(webcc::RequestPtr request);
|
||||||
|
|
||||||
|
// Update a book.
|
||||||
|
webcc::ResponsePtr Put(webcc::RequestPtr request);
|
||||||
|
|
||||||
|
// Delete a book.
|
||||||
|
webcc::ResponsePtr Delete(webcc::RequestPtr request);
|
||||||
|
|
||||||
|
private:
|
||||||
|
bfs::path photo_dir_;
|
||||||
|
};
|
||||||
|
|
||||||
|
// -----------------------------------------------------------------------------
|
||||||
|
|
||||||
|
// URL: /books/{id}/photo
|
||||||
|
class BookPhotoView : public webcc::View {
|
||||||
|
public:
|
||||||
|
explicit BookPhotoView(bfs::path photo_dir);
|
||||||
|
|
||||||
|
webcc::ResponsePtr Handle(webcc::RequestPtr request) override;
|
||||||
|
|
||||||
|
// Stream the request data, an image, of PUT into a temp file.
|
||||||
|
bool Stream(const std::string& method) override {
|
||||||
|
return method == "PUT";
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
// Get the photo of the book.
|
||||||
|
webcc::ResponsePtr Get(webcc::RequestPtr request);
|
||||||
|
|
||||||
|
// Set the photo of the book.
|
||||||
|
webcc::ResponsePtr Put(webcc::RequestPtr request);
|
||||||
|
|
||||||
|
// Delete the photo of the book.
|
||||||
|
webcc::ResponsePtr Delete(webcc::RequestPtr request);
|
||||||
|
|
||||||
|
private:
|
||||||
|
bfs::path photo_dir_;
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif // VIEWS_H_
|
@ -1,286 +0,0 @@
|
|||||||
#include <iostream>
|
|
||||||
#include <string>
|
|
||||||
|
|
||||||
#include "boost/filesystem/operations.hpp"
|
|
||||||
|
|
||||||
#include "json/json.h"
|
|
||||||
|
|
||||||
#include "webcc/logger.h"
|
|
||||||
#include "webcc/response_builder.h"
|
|
||||||
#include "webcc/server.h"
|
|
||||||
|
|
||||||
#include "examples/common/book.h"
|
|
||||||
#include "examples/common/book_json.h"
|
|
||||||
|
|
||||||
#if (defined(_WIN32) || defined(_WIN64))
|
|
||||||
#if defined(_DEBUG) && defined(WEBCC_ENABLE_VLD)
|
|
||||||
#pragma message ("< include vld.h >")
|
|
||||||
#include "vld/vld.h"
|
|
||||||
#pragma comment(lib, "vld")
|
|
||||||
#endif
|
|
||||||
#endif
|
|
||||||
|
|
||||||
namespace bfs = boost::filesystem;
|
|
||||||
|
|
||||||
// -----------------------------------------------------------------------------
|
|
||||||
|
|
||||||
static BookStore g_book_store;
|
|
||||||
|
|
||||||
// -----------------------------------------------------------------------------
|
|
||||||
// BookListView
|
|
||||||
|
|
||||||
// URL: /books
|
|
||||||
class BookListView : public webcc::View {
|
|
||||||
public:
|
|
||||||
webcc::ResponsePtr Handle(webcc::RequestPtr request) override {
|
|
||||||
if (request->method() == "GET") {
|
|
||||||
return Get(request);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (request->method() == "POST") {
|
|
||||||
return Post(request);
|
|
||||||
}
|
|
||||||
|
|
||||||
return {};
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
|
||||||
// Get a list of books based on query parameters.
|
|
||||||
webcc::ResponsePtr Get(webcc::RequestPtr request) {
|
|
||||||
Json::Value json(Json::arrayValue);
|
|
||||||
|
|
||||||
for (const Book& book : g_book_store.books()) {
|
|
||||||
json.append(BookToJson(book));
|
|
||||||
}
|
|
||||||
|
|
||||||
// Return all books as a JSON array.
|
|
||||||
|
|
||||||
return webcc::ResponseBuilder{}.OK().Body(JsonToString(json)).Json().Utf8()();
|
|
||||||
}
|
|
||||||
|
|
||||||
// Create a new book.
|
|
||||||
webcc::ResponsePtr Post(webcc::RequestPtr request) {
|
|
||||||
Book book;
|
|
||||||
if (JsonStringToBook(request->data(), &book)) {
|
|
||||||
std::string id = g_book_store.AddBook(book);
|
|
||||||
|
|
||||||
Json::Value json;
|
|
||||||
json["id"] = id;
|
|
||||||
|
|
||||||
return webcc::ResponseBuilder{}.Created().Body(JsonToString(json)).Json().Utf8()();
|
|
||||||
} else {
|
|
||||||
// Invalid JSON
|
|
||||||
return webcc::ResponseBuilder{}.BadRequest()();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
// -----------------------------------------------------------------------------
|
|
||||||
// BookDetailView
|
|
||||||
|
|
||||||
// URL: /books/{id}
|
|
||||||
class BookDetailView : public webcc::View {
|
|
||||||
public:
|
|
||||||
webcc::ResponsePtr Handle(webcc::RequestPtr request) override {
|
|
||||||
if (request->method() == "GET") {
|
|
||||||
return Get(request);
|
|
||||||
}
|
|
||||||
if (request->method() == "PUT") {
|
|
||||||
return Put(request);
|
|
||||||
}
|
|
||||||
if (request->method() == "DELETE") {
|
|
||||||
return Delete(request);
|
|
||||||
}
|
|
||||||
return {};
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
|
||||||
// Get the detailed information of a book.
|
|
||||||
webcc::ResponsePtr Get(webcc::RequestPtr request) {
|
|
||||||
if (request->args().size() != 1) {
|
|
||||||
// NotFound means the resource specified by the URL cannot be found.
|
|
||||||
// BadRequest could be another choice.
|
|
||||||
return webcc::ResponseBuilder{}.NotFound()();
|
|
||||||
}
|
|
||||||
|
|
||||||
const std::string& id = request->args()[0];
|
|
||||||
|
|
||||||
const Book& book = g_book_store.GetBook(id);
|
|
||||||
if (book.IsNull()) {
|
|
||||||
return webcc::ResponseBuilder{}.NotFound()();
|
|
||||||
}
|
|
||||||
|
|
||||||
return webcc::ResponseBuilder{}.OK().Body(BookToJsonString(book)).Json().Utf8()();
|
|
||||||
}
|
|
||||||
|
|
||||||
// Update a book.
|
|
||||||
webcc::ResponsePtr Put(webcc::RequestPtr request) {
|
|
||||||
if (request->args().size() != 1) {
|
|
||||||
return webcc::ResponseBuilder{}.NotFound()();
|
|
||||||
}
|
|
||||||
|
|
||||||
const std::string& id = request->args()[0];
|
|
||||||
|
|
||||||
Book book;
|
|
||||||
if (!JsonStringToBook(request->data(), &book)) {
|
|
||||||
return webcc::ResponseBuilder{}.BadRequest()();
|
|
||||||
}
|
|
||||||
|
|
||||||
book.id = id;
|
|
||||||
g_book_store.UpdateBook(book);
|
|
||||||
|
|
||||||
return webcc::ResponseBuilder{}.OK()();
|
|
||||||
}
|
|
||||||
|
|
||||||
// Delete a book.
|
|
||||||
webcc::ResponsePtr Delete(webcc::RequestPtr request) {
|
|
||||||
if (request->args().size() != 1) {
|
|
||||||
return webcc::ResponseBuilder{}.NotFound()();
|
|
||||||
}
|
|
||||||
|
|
||||||
const std::string& id = request->args()[0];
|
|
||||||
|
|
||||||
if (!g_book_store.DeleteBook(id)) {
|
|
||||||
return webcc::ResponseBuilder{}.NotFound()();
|
|
||||||
}
|
|
||||||
|
|
||||||
return webcc::ResponseBuilder{}.OK()();
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
// -----------------------------------------------------------------------------
|
|
||||||
// BookPhotoView
|
|
||||||
|
|
||||||
// URL: /books/{id}/photo
|
|
||||||
class BookPhotoView : public webcc::View {
|
|
||||||
public:
|
|
||||||
explicit BookPhotoView(bfs::path upload_dir)
|
|
||||||
: upload_dir_(std::move(upload_dir)) {
|
|
||||||
}
|
|
||||||
|
|
||||||
webcc::ResponsePtr Handle(webcc::RequestPtr request) override {
|
|
||||||
if (request->method() == "GET") {
|
|
||||||
return Get(request);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (request->method() == "PUT") {
|
|
||||||
return Put(request);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (request->method() == "DELETE") {
|
|
||||||
return Delete(request);
|
|
||||||
}
|
|
||||||
|
|
||||||
return {};
|
|
||||||
}
|
|
||||||
|
|
||||||
// Stream the request data, an image, of PUT into a temp file.
|
|
||||||
bool Stream(const std::string& method) override {
|
|
||||||
return method == "PUT";
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
|
||||||
// Get the photo of the book.
|
|
||||||
// TODO: Check content type to see if it's JPEG.
|
|
||||||
webcc::ResponsePtr Get(webcc::RequestPtr request) {
|
|
||||||
if (request->args().size() != 1) {
|
|
||||||
return webcc::ResponseBuilder{}.NotFound()();
|
|
||||||
}
|
|
||||||
|
|
||||||
const std::string& id = request->args()[0];
|
|
||||||
const Book& book = g_book_store.GetBook(id);
|
|
||||||
if (book.IsNull()) {
|
|
||||||
return webcc::ResponseBuilder{}.NotFound()();
|
|
||||||
}
|
|
||||||
|
|
||||||
bfs::path photo_path = GetPhotoPath(id);
|
|
||||||
if (!bfs::exists(photo_path)) {
|
|
||||||
return webcc::ResponseBuilder{}.NotFound()();
|
|
||||||
}
|
|
||||||
|
|
||||||
// File() might throw Error::kFileError.
|
|
||||||
// TODO: Avoid exception handling.
|
|
||||||
try {
|
|
||||||
return webcc::ResponseBuilder{}.OK().File(photo_path)();
|
|
||||||
} catch (const webcc::Error&) {
|
|
||||||
return webcc::ResponseBuilder{}.NotFound()();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Set the photo of the book.
|
|
||||||
// TODO: Check content type to see if it's JPEG.
|
|
||||||
webcc::ResponsePtr Put(webcc::RequestPtr request) {
|
|
||||||
if (request->args().size() != 1) {
|
|
||||||
return webcc::ResponseBuilder{}.NotFound()();
|
|
||||||
}
|
|
||||||
|
|
||||||
const std::string& id = request->args()[0];
|
|
||||||
|
|
||||||
const Book& book = g_book_store.GetBook(id);
|
|
||||||
if (book.IsNull()) {
|
|
||||||
return webcc::ResponseBuilder{}.NotFound()();
|
|
||||||
}
|
|
||||||
|
|
||||||
request->file_body()->Move(GetPhotoPath(id));
|
|
||||||
|
|
||||||
return webcc::ResponseBuilder{}.OK()();
|
|
||||||
}
|
|
||||||
|
|
||||||
// Delete the photo of the book.
|
|
||||||
webcc::ResponsePtr Delete(webcc::RequestPtr request) {
|
|
||||||
return {};
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
|
||||||
bfs::path GetPhotoPath(const std::string& book_id) const {
|
|
||||||
return upload_dir_ / "book_photo" / (book_id + ".jpg");
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
|
||||||
bfs::path upload_dir_;
|
|
||||||
};
|
|
||||||
|
|
||||||
// -----------------------------------------------------------------------------
|
|
||||||
|
|
||||||
int main(int argc, char* argv[]) {
|
|
||||||
if (argc < 3) {
|
|
||||||
std::cout << "usage: rest_book_server <port> <upload_dir>" << std::endl;
|
|
||||||
std::cout << "examples:" << std::endl;
|
|
||||||
std::cout << " $ rest_book_server 8080 D:/upload" << std::endl;
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
WEBCC_LOG_INIT("", webcc::LOG_CONSOLE);
|
|
||||||
|
|
||||||
std::uint16_t port = static_cast<std::uint16_t>(std::atoi(argv[1]));
|
|
||||||
|
|
||||||
bfs::path upload_dir = argv[2];
|
|
||||||
if (!bfs::is_directory(upload_dir) || !bfs::exists(upload_dir)) {
|
|
||||||
std::cerr << "Invalid upload dir!" << std::endl;
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
try {
|
|
||||||
webcc::Server server(port); // No doc root
|
|
||||||
|
|
||||||
server.Route("/books",
|
|
||||||
std::make_shared<BookListView>(),
|
|
||||||
{ "GET", "POST" });
|
|
||||||
|
|
||||||
server.Route(webcc::R("/books/(\\d+)"),
|
|
||||||
std::make_shared<BookDetailView>(),
|
|
||||||
{ "GET", "PUT", "DELETE" });
|
|
||||||
|
|
||||||
server.Route(webcc::R("/books/(\\d+)/photo"),
|
|
||||||
std::make_shared<BookPhotoView>(upload_dir),
|
|
||||||
{ "GET", "PUT", "DELETE" });
|
|
||||||
|
|
||||||
server.Run(2);
|
|
||||||
|
|
||||||
} catch (const std::exception& e) {
|
|
||||||
std::cerr << e.what() << std::endl;
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
Loading…
Reference in New Issue