57constexpr char const *gSQliteVfsName = 
"ROOT-Davix-readonly";
 
   68   VfsRootFile() : pos(&
c) {}
 
   81int VfsRdOnlyClose(sqlite3_file *pFile)
 
   83   Davix::DavixError *err = 
nullptr;
 
   84   VfsRootFile *p = 
reinterpret_cast<VfsRootFile *
>(pFile);
 
   85   auto retval = p->pos.close(p->fd, &err);
 
   88   return (retval == -1) ? SQLITE_IOERR_CLOSE : SQLITE_OK;
 
   93int VfsRdOnlyRead(sqlite3_file *pFile, 
void *zBuf, 
int count, sqlite_int64 offset)
 
   95   Davix::DavixError *err = 
nullptr;
 
   96   VfsRootFile *p = 
reinterpret_cast<VfsRootFile *
>(pFile);
 
   97   auto retval = p->pos.pread(p->fd, zBuf, count, offset, &err);
 
   98   return (retval == -1) ? SQLITE_IOERR : SQLITE_OK;
 
  103int VfsRdOnlyWrite(sqlite3_file * , 
const void * , 
int , sqlite_int64 )
 
  105   return SQLITE_OPEN_READONLY;
 
  110int VfsRdOnlyTruncate(sqlite3_file * , sqlite_int64 )
 
  112   return SQLITE_OPEN_READONLY;
 
  117int VfsRdOnlySync(sqlite3_file * , 
int )
 
  124int VfsRdOnlyFileSize(sqlite3_file *pFile, sqlite_int64 *pSize)
 
  126   VfsRootFile *p = 
reinterpret_cast<VfsRootFile *
>(pFile);
 
  133int VfsRdOnlyLock(sqlite3_file * , 
int )
 
  140int VfsRdOnlyUnlock(sqlite3_file * , 
int )
 
  147int VfsRdOnlyCheckReservedLock(sqlite3_file * , 
int *pResOut)
 
  155int VfsRdOnlyFileControl(sqlite3_file * , 
int , 
void * )
 
  157   return SQLITE_NOTFOUND;
 
  162int VfsRdOnlySectorSize(sqlite3_file * )
 
  164   return SQLITE_OPEN_READONLY;
 
  169int VfsRdOnlyDeviceCharacteristics(sqlite3_file * )
 
  171   return SQLITE_OPEN_READONLY;
 
  176int VfsRdOnlyOpen(sqlite3_vfs * , 
const char *zName, sqlite3_file *pFile, 
int flags, 
int * )
 
  179   VfsRootFile *p = 
new (pFile) VfsRootFile();
 
  180   p->pFile.pMethods = 
nullptr;
 
  184   static const sqlite3_io_methods io_methods = {
 
  194      VfsRdOnlyCheckReservedLock,
 
  195      VfsRdOnlyFileControl,
 
  197      VfsRdOnlyDeviceCharacteristics,
 
  207   if (flags & (SQLITE_OPEN_READWRITE | SQLITE_OPEN_DELETEONCLOSE | SQLITE_OPEN_EXCLUSIVE))
 
  210   Davix::DavixError *err = 
nullptr;
 
  211   p->fd = p->pos.open(
nullptr, zName, O_RDONLY, &err);
 
  214      ::Error(
"VfsRdOnlyOpen", 
"%s\n", err->getErrMsg().c_str());
 
  219   if (p->pos.stat(
nullptr, zName, &buf, 
nullptr) == -1) {
 
  222   p->size = buf.st_size;
 
  224   p->pFile.pMethods = &io_methods;
 
  234int VfsRdOnlyDelete(sqlite3_vfs * , 
const char * , 
int )
 
  236   return SQLITE_IOERR_DELETE;
 
  241int VfsRdOnlyAccess(sqlite3_vfs * , 
const char * , 
int flags, 
int *pResOut)
 
  244   if (flags == SQLITE_ACCESS_READWRITE) {
 
  245      return SQLITE_OPEN_READONLY;
 
  252int VfsRdOnlyFullPathname(sqlite3_vfs * , 
const char *zPath, 
int nOut, 
char *zOut)
 
  254   zOut[nOut - 1] = 
'\0';
 
  255   sqlite3_snprintf(nOut, zOut, 
"%s", zPath);
 
  261int VfsRdOnlyRandomness(sqlite3_vfs * , 
int nBuf, 
char *zBuf)
 
  263   for (
int i = 0; i < nBuf; ++i) {
 
  271int VfsRdOnlySleep(sqlite3_vfs * , 
int microseconds)
 
  280int VfsRdOnlyGetLastError(sqlite3_vfs * , 
int , 
char * )
 
  287int VfsRdOnlyCurrentTimeInt64(sqlite3_vfs * , sqlite3_int64 *piNow)
 
  289   static constexpr sqlite3_int64 unixEpoch = 24405875 * (sqlite3_int64)8640000;
 
  292   *piNow = ((sqlite3_int64)t) * 1000 + unixEpoch;
 
  298int VfsRdOnlyCurrentTime(sqlite3_vfs *vfs, 
double *prNow)
 
  301   int rc = VfsRdOnlyCurrentTimeInt64(vfs, &i);
 
  302   *prNow = i / 86400000.0;
 
  308static struct sqlite3_vfs kSqlite3Vfs = {
 
  318   VfsRdOnlyFullPathname,
 
  325   VfsRdOnlyCurrentTime,
 
  326   VfsRdOnlyGetLastError,
 
  327   VfsRdOnlyCurrentTimeInt64,
 
  336bool RegisterDavixVfs()
 
  340   retval = sqlite3_vfs_register(&kSqlite3Vfs, 
false);
 
  341   return (retval == SQLITE_OK);
 
  347bool IsURL(
const std::string &fileName)
 
  349   if (fileName.compare(0, 7, 
"http://") == 0)
 
  351   if (fileName.compare(0, 8, 
"https://") == 0)
 
  365struct RSqliteDSDataSet {
 
  366   sqlite3 *fDb = 
nullptr;
 
  367   sqlite3_stmt *fQuery = 
nullptr;
 
  372   : 
fType(
type), fIsActive(false), fInteger(0), fReal(0.0), fText(), fBlob(), fNull(nullptr)
 
  380   default: 
throw std::runtime_error(
"Internal error");
 
  395   static bool isDavixAvailable = RegisterDavixVfs();
 
  399   if (IsURL(fileName)) {
 
  400      if (!isDavixAvailable)
 
  401         throw std::runtime_error(
"Processing remote files is not available. " 
  402                                  "Please compile ROOT with Davix support to read from HTTP(S) locations.");
 
  404        sqlite3_open_v2(fileName.c_str(), &
fDataSet->fDb, SQLITE_OPEN_READONLY | SQLITE_OPEN_NOMUTEX, gSQliteVfsName);
 
  406      retval = sqlite3_open_v2(fileName.c_str(), &
fDataSet->fDb, SQLITE_OPEN_READONLY | SQLITE_OPEN_NOMUTEX, 
nullptr);
 
  408   if (retval != SQLITE_OK)
 
  411   retval = sqlite3_prepare_v2(
fDataSet->fDb, query.c_str(), -1, &
fDataSet->fQuery, 
nullptr);
 
  412   if (retval != SQLITE_OK)
 
  415   int colCount = sqlite3_column_count(
fDataSet->fQuery);
 
  416   retval = sqlite3_step(
fDataSet->fQuery);
 
  417   if ((retval != SQLITE_ROW) && (retval != SQLITE_DONE))
 
  421   for (
int i = 0; i < colCount; ++i) {
 
  423      int type = SQLITE_NULL;
 
  426      const char *declTypeCstr = sqlite3_column_decltype(
fDataSet->fQuery, i);
 
  427      if (declTypeCstr == 
nullptr) {
 
  428         if (retval == SQLITE_ROW)
 
  431         std::string declType(declTypeCstr);
 
  432         std::transform(declType.begin(), declType.end(), declType.begin(), ::toupper);
 
  433         if (declType == 
"INTEGER")
 
  434            type = SQLITE_INTEGER;
 
  435         else if (declType == 
"FLOAT")
 
  437         else if (declType == 
"TEXT")
 
  439         else if (declType == 
"BLOB")
 
  442            throw std::runtime_error(
"Unexpected column decl type");
 
  467      default: 
throw std::runtime_error(
"Unhandled data type");
 
  503      std::string errmsg = 
"The type selected for column \"";
 
  505      errmsg += 
"\" does not correspond to column type, which is ";
 
  507      throw std::runtime_error(errmsg);
 
  510   fValues[index].fIsActive = 
true;
 
  519   std::vector<std::pair<ULong64_t, ULong64_t>> entryRanges;
 
  520   int retval = sqlite3_step(
fDataSet->fQuery);
 
  522   case SQLITE_DONE: 
return entryRanges;
 
  540   for (
unsigned i = 0; i < 
N; ++i) {
 
  545   throw std::runtime_error(
"Unknown column: " + std::string(colName));
 
  560   int retval = sqlite3_reset(
fDataSet->fQuery);
 
  561   if (retval != SQLITE_OK)
 
  562      throw std::runtime_error(
"SQlite error, reset");
 
  576   ROOT::RDataFrame rdf(std::make_unique<RSqliteDS>(std::string(fileName), std::string(query)));
 
  586   for (
unsigned i = 0; i < 
N; ++i) {
 
  595         nbytes = sqlite3_column_bytes(
fDataSet->fQuery, i);
 
  599            fValues[i].fText = 
reinterpret_cast<const char *
>(sqlite3_column_text(
fDataSet->fQuery, i));
 
  603         nbytes = sqlite3_column_bytes(
fDataSet->fQuery, i);
 
  604         fValues[i].fBlob.resize(nbytes);
 
  606            std::memcpy(
fValues[i].fBlob.data(), sqlite3_column_blob(
fDataSet->fQuery, i), nbytes);
 
  610      default: 
throw std::runtime_error(
"Unhandled column type");
 
  621      ::Warning(
"SetNSlots", 
"Currently the SQlite data source faces performance degradation in multi-threaded mode. " 
  622                             "Consider turning off IMT.");
 
  631   std::string errmsg = 
"SQlite error: ";
 
  632   errmsg += sqlite3_errstr(errcode);
 
  633   throw std::runtime_error(errmsg);
 
unsigned long long ULong64_t
void Error(const char *location, const char *msgfmt,...)
void Warning(const char *location, const char *msgfmt,...)
R__EXTERN TRandom * gRandom
R__EXTERN TSystem * gSystem
std::vector< void * > Record_t
void SetNSlots(unsigned int nSlots) final
Almost a no-op, many slots can in fact reduce the performance due to thread synchronization.
static constexpr char const  * fgTypeNames[]
Corresponds to the types defined in ETypes.
std::string GetLabel() final
Return a string representation of the datasource type.
void Initialise() final
Resets the SQlite query engine at the beginning of the event loop.
std::vector< std::string > fColumnNames
~RSqliteDS()
Frees the sqlite resources and closes the file.
bool HasColumn(std::string_view colName) const final
A linear search through the columns for the given name.
std::vector< ETypes > fColumnTypes
std::string GetTypeName(std::string_view colName) const final
Returns the C++ type for a given column name, implemented as a linear search through all the columns.
ETypes
All the types known to SQlite. Changes require changing fgTypeNames, too.
Record_t GetColumnReadersImpl(std::string_view name, const std::type_info &) final
Activates the given column's result value.
RSqliteDS(const std::string &fileName, const std::string &query)
Build the dataframe.
std::unique_ptr< Internal::RSqliteDSDataSet > fDataSet
std::vector< std::pair< ULong64_t, ULong64_t > > GetEntryRanges() final
Returns a range of size 1 as long as more rows are available in the SQL result set.
const std::vector< std::string > & GetColumnNames() const final
Returns the SELECT queries names.
bool SetEntry(unsigned int slot, ULong64_t entry) final
Stores the result of the current active sqlite query row as a C++ value.
void SqliteError(int errcode)
Helper function to throw an exception if there is a fatal sqlite error, e.g. an I/O error.
std::vector< Value_t > fValues
The data source is inherently single-threaded and returns only one row at a time. This vector holds t...
ROOT's RDataFrame offers a high level interface for analyses of data stored in TTrees,...
virtual UInt_t Integer(UInt_t imax)
Returns a random integer on [ 0, imax-1 ].
virtual void Sleep(UInt_t milliSec)
Sleep milliSec milli seconds.
RDataFrame MakeSqliteDataFrame(std::string_view fileName, std::string_view query)
Factory method to create a SQlite RDataFrame.
Namespace for new ROOT classes and functions.
std::unique_ptr< T > make_unique(Args &&... args)
basic_string_view< char > string_view
void * fPtr
Points to one of the values; an address to this pointer is returned by GetColumnReadersImpl.
std::vector< unsigned char > fBlob