#pragma once #include #include #include #include #include #include #include #include #include #include #include #include namespace DB { class StorageFileLog final : public shared_ptr_helper, public IStorage, WithContext { friend struct shared_ptr_helper; public: enum class FileStatus { BEGIN, NO_CHANGE, UPDATED, REMOVED }; using Files = std::vector; std::string getName() const override { return "FileLog"; } bool noPushingToViews() const override { return true; } void startup() override; void shutdown() override; Pipe read( const Names & column_names, const StorageMetadataPtr & /*metadata_snapshot*/, SelectQueryInfo & query_info, ContextPtr context, QueryProcessingStage::Enum processed_stage, size_t max_block_size, unsigned num_streams) override; const auto & getFormatName() const { return format_name; } auto & getFileNames() { return file_names; } auto & getFileStatuses() { return file_statuses; } protected: StorageFileLog( const StorageID & table_id_, ContextPtr context_, const ColumnsDescription & columns_, const String & relative_path_, const String & format_name_, std::unique_ptr settings); private: std::unique_ptr filelog_settings; const String path; bool path_is_directory = false; const String format_name; Poco::Logger * log; struct FileContext { FileStatus status = FileStatus::BEGIN; size_t last_read_position = 0; }; using NameToFile = std::unordered_map; NameToFile file_statuses; std::vector file_names; std::mutex status_mutex; std::unique_ptr directory_watch = nullptr; uint64_t milliseconds_to_wait; // Stream thread struct TaskContext { BackgroundSchedulePool::TaskHolder holder; std::atomic stream_cancelled {false}; explicit TaskContext(BackgroundSchedulePool::TaskHolder&& task_) : holder(std::move(task_)) { } }; std::shared_ptr task; using TaskThread = BackgroundSchedulePool::TaskHolder; void threadFunc(); size_t getPollMaxBatchSize() const; size_t getMaxBlockSize() const; size_t getPollTimeoutMillisecond() const; bool streamToViews(); bool checkDependencies(const StorageID & table_id); bool updateFileStatuses(); }; }