| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177 |
- #include "FileCache.h"
- #include "herr.h"
- #include "hscope.h"
- #include "htime.h"
- #include "hlog.h"
- #include "httpdef.h" // import http_content_type_str_by_suffix
- #include "http_page.h" // import make_index_of_page
- #ifdef OS_WIN
- #include <codecvt>
- #endif
- #define ETAG_FMT "\"%zx-%zx\""
- FileCache::FileCache() {
- stat_interval = 10; // s
- expired_time = 60; // s
- }
- static int hv_open(char const* filepath, int flags) {
- #ifdef OS_WIN
- std::wstring_convert<std::codecvt_utf8_utf16<wchar_t>> conv;
- auto wfilepath = conv.from_bytes(filepath);
- int fd = _wopen(wfilepath.c_str(), flags);
- #else
- int fd = open(filepath, flags);
- #endif
- return fd;
- }
- file_cache_ptr FileCache::Open(const char* filepath, OpenParam* param) {
- std::lock_guard<std::mutex> locker(mutex_);
- file_cache_ptr fc = Get(filepath);
- bool modified = false;
- if (fc) {
- time_t now = time(NULL);
- if (now - fc->stat_time > stat_interval) {
- modified = fc->is_modified();
- fc->stat_time = now;
- fc->stat_cnt++;
- }
- if (param->need_read) {
- if (!modified && fc->is_complete()) {
- param->need_read = false;
- }
- }
- }
- if (fc == NULL || modified || param->need_read) {
- int flags = O_RDONLY;
- #ifdef O_BINARY
- flags |= O_BINARY;
- #endif
- int fd = hv_open(filepath, flags);
- if (fd < 0) {
- #ifdef OS_WIN
- // NOTE: open(dir) return -1 on windows
- if (!hv_isdir(filepath)) {
- param->error = ERR_OPEN_FILE;
- return NULL;
- }
- #else
- param->error = ERR_OPEN_FILE;
- return NULL;
- #endif
- }
- defer(if (fd > 0) { close(fd); })
- if (fc == NULL) {
- struct stat st;
- if (fd > 0) {
- fstat(fd, &st);
- } else {
- stat(filepath, &st);
- }
- if (S_ISREG(st.st_mode) ||
- (S_ISDIR(st.st_mode) &&
- filepath[strlen(filepath)-1] == '/')) {
- fc = std::make_shared<file_cache_t>();
- fc->filepath = filepath;
- fc->st = st;
- time(&fc->open_time);
- fc->stat_time = fc->open_time;
- fc->stat_cnt = 1;
- cached_files[filepath] = fc;
- }
- else {
- param->error = ERR_MISMATCH;
- return NULL;
- }
- }
- if (S_ISREG(fc->st.st_mode)) {
- param->filesize = fc->st.st_size;
- // FILE
- if (param->need_read) {
- if (fc->st.st_size > param->max_read) {
- param->error = ERR_OVER_LIMIT;
- return NULL;
- }
- fc->resize_buf(fc->st.st_size);
- int nread = read(fd, fc->filebuf.base, fc->filebuf.len);
- if (nread != fc->filebuf.len) {
- hloge("Failed to read file: %s", filepath);
- param->error = ERR_READ_FILE;
- return NULL;
- }
- }
- const char* suffix = strrchr(filepath, '.');
- if (suffix) {
- http_content_type content_type = http_content_type_enum_by_suffix(suffix+1);
- if (content_type == TEXT_HTML) {
- fc->content_type = "text/html; charset=utf-8";
- } else if (content_type == TEXT_PLAIN) {
- fc->content_type = "text/plain; charset=utf-8";
- } else {
- fc->content_type = http_content_type_str_by_suffix(suffix+1);
- }
- }
- }
- else if (S_ISDIR(fc->st.st_mode)) {
- // DIR
- std::string page;
- make_index_of_page(filepath, page, param->path);
- fc->resize_buf(page.size());
- memcpy(fc->filebuf.base, page.c_str(), page.size());
- fc->content_type = "text/html; charset=utf-8";
- }
- gmtime_fmt(fc->st.st_mtime, fc->last_modified);
- snprintf(fc->etag, sizeof(fc->etag), ETAG_FMT, (size_t)fc->st.st_mtime, (size_t)fc->st.st_size);
- }
- return fc;
- }
- bool FileCache::Close(const char* filepath) {
- std::lock_guard<std::mutex> locker(mutex_);
- auto iter = cached_files.find(filepath);
- if (iter != cached_files.end()) {
- iter = cached_files.erase(iter);
- return true;
- }
- return false;
- }
- bool FileCache::Close(const file_cache_ptr& fc) {
- std::lock_guard<std::mutex> locker(mutex_);
- auto iter = cached_files.begin();
- while (iter != cached_files.end()) {
- if (iter->second == fc) {
- iter = cached_files.erase(iter);
- return true;
- } else {
- ++iter;
- }
- }
- return false;
- }
- file_cache_ptr FileCache::Get(const char* filepath) {
- auto iter = cached_files.find(filepath);
- if (iter != cached_files.end()) {
- return iter->second;
- }
- return NULL;
- }
- void FileCache::RemoveExpiredFileCache() {
- std::lock_guard<std::mutex> locker(mutex_);
- time_t now = time(NULL);
- auto iter = cached_files.begin();
- while (iter != cached_files.end()) {
- if (now - iter->second->stat_time > expired_time) {
- iter = cached_files.erase(iter);
- } else {
- ++iter;
- }
- }
- }
|