#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 _MSC_VER #include #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 _MSC_VER std::wstring_convert> 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 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(); 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 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 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 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; } } }