takane
Validators for ArtifactDB file formats
Loading...
Searching...
No Matches
gff_file.hpp
Go to the documentation of this file.
1#ifndef TAKANE_GFF_FILE_HPP
2#define TAKANE_GFF_FILE_HPP
3
4#include "utils_files.hpp"
5#include "ritsuko/ritsuko.hpp"
6
7#include <filesystem>
8#include <stdexcept>
9#include <string>
10
16namespace takane {
17
22namespace gff_file {
23
32inline void validate(const std::filesystem::path& path, const ObjectMetadata& metadata, Options& options) {
33 const auto& gffmap = internal_json::extract_typed_object_from_metadata(metadata.other, "gff_file");
34
35 const std::string& vstring = internal_json::extract_string_from_typed_object(gffmap, "version", "gff_file");
36 auto version = ritsuko::parse_version_string(vstring.c_str(), vstring.size(), /* skip_patch = */ true);
37 if (version.major != 1) {
38 throw std::runtime_error("unsupported version string '" + vstring + "'");
39 }
40
41 auto fpath = path / "file.";
42
43 const std::string& fstring = internal_json::extract_string_from_typed_object(gffmap, "format", "gff_file");
44 if (fstring == "GFF2") {
45 fpath += "gff2";
46 } else if (fstring == "GFF3") {
47 fpath += "gff3";
48 } else {
49 throw std::runtime_error("unknown value '" + fstring + "' for 'gff_file.format' property");
50 }
51
52 // Check if it's indexed.
53 bool indexed = internal_files::is_indexed(gffmap);
54 fpath += ".";
55 if (indexed) {
56 fpath += "bgz";
57 } else {
58 fpath += "gz";
59 }
60
61 // Check magic numbers.
62 internal_files::check_gzip_signature(fpath);
63
64 if (fstring == "GFF3") {
65 const std::string expected = "##gff-version 3";
66 const size_t expected_len = expected.size();
67
68 auto reader = internal_other::open_reader<byteme::GzipFileReader>(fpath, expected_len);
69 byteme::PerByte<> pb(&reader);
70 bool okay = pb.valid();
71
72 for (size_t i = 0; i < expected_len; ++i) {
73 if (!okay) {
74 throw std::runtime_error("incomplete GFF3 file signature for '" + fpath.string() + "'");
75 }
76 if (pb.get() != expected[i]) {
77 throw std::runtime_error("incorrect GFF3 file signature for '" + fpath.string() + "'");
78 }
79 okay = pb.advance();
80 }
81 }
82
83 if (indexed) {
84 auto ixpath = fpath;
85 ixpath += ".tbi";
86 internal_files::check_gzip_signature(ixpath);
87 internal_files::check_signature<byteme::GzipFileReader>(ixpath, "TBI\1", 4, "tabix");
88 }
89
90 if (options.gff_file_strict_check) {
91 options.gff_file_strict_check(path, metadata, options, indexed);
92 }
93}
94
95}
96
97}
98
99#endif
void validate(const std::filesystem::path &path, const ObjectMetadata &metadata, Options &options)
Definition gff_file.hpp:32
takane validation functions.
Definition _derived_from.hpp:15
Object metadata, including the type and other fields.
Definition utils_public.hpp:26
std::unordered_map< std::string, std::shared_ptr< millijson::Base > > other
Definition utils_public.hpp:35
Validation options.
Definition utils_public.hpp:94
std::function< void(const std::filesystem::path &, const ObjectMetadata &, Options &, bool)> gff_file_strict_check
Definition utils_public.hpp:215