HEX
Server: Apache
System: Linux wp02.tdr-lab.com 3.10.0-1160.42.2.el7.x86_64 #1 SMP Tue Sep 7 14:49:57 UTC 2021 x86_64
User: kusanagi (1001)
PHP: 7.4.23
Disabled: NONE
Upload Files
File: //proc/self/root/usr/include/hphp/runtime/vm/repo-helpers.h
/*
   +----------------------------------------------------------------------+
   | HipHop for PHP                                                       |
   +----------------------------------------------------------------------+
   | Copyright (c) 2010-present Facebook, Inc. (http://www.facebook.com)  |
   +----------------------------------------------------------------------+
   | This source file is subject to version 3.01 of the PHP license,      |
   | that is bundled with this package in the file LICENSE, and is        |
   | available through the world-wide-web at the following url:           |
   | http://www.php.net/license/3_01.txt                                  |
   | If you did not receive a copy of the PHP license and are unable to   |
   | obtain it through the world-wide-web, please send a note to          |
   | license@php.net so we can mail you a copy immediately.               |
   +----------------------------------------------------------------------+
*/

#ifndef incl_HPHP_VM_REPO_HELPERS_H_
#define incl_HPHP_VM_REPO_HELPERS_H_

#include "hphp/runtime/base/attr.h"
#include "hphp/runtime/base/types.h"

#include "hphp/util/md5.h"
#include "hphp/util/portability.h"

#include <sqlite3.h>

#include <folly/portability/Stdio.h>

namespace HPHP {

// Forward declaration.
struct BlobDecoder;
struct BlobEncoder;
struct StringData;
struct TypedValue;
struct Repo;

enum RepoId {
  RepoIdInvalid = -1,
  RepoIdCentral =  0,
  RepoIdLocal   =  1,

  RepoIdCount   =  2 // Number of database IDs.
};

struct RepoExc : std::exception {
  RepoExc(ATTRIBUTE_PRINTF_STRING const char* fmt, ...)
    ATTRIBUTE_PRINTF(2, 3) {
    va_list(ap);
    va_start(ap, fmt);
    char* msg;
    if (vasprintf(&msg, fmt, ap) == -1) {
      m_msg = "";
    } else {
      m_msg = msg;
      free(msg);
    }
    va_end(ap);
  }
  const std::string& msg() const { return m_msg; }
  const char* what() const noexcept override { return m_msg.c_str(); }

 private:
  std::string m_msg;
};

struct RepoStmt {
  explicit RepoStmt(Repo& repo);
  ~RepoStmt();
 private:
  void finalize();
 public:
  bool prepared() const { return (m_stmt != nullptr); }
  void prepare(const std::string& sql); // throws(RepoExc)
  void reset();
  Repo& repo() const { return m_repo; }
  sqlite3_stmt*& get() { return m_stmt; }
  const std::string& sql() const { return m_sql; }

 protected:
  Repo& m_repo;
  std::string m_sql;
  sqlite3_stmt* m_stmt;
};

// Under most circumstances RepoTxnQuery should be used instead of RepoQuery;
// queries outside of transactions are fraught with peril.
struct RepoQuery {
  explicit RepoQuery(RepoStmt& stmt)
    : m_stmt(stmt), m_row(false), m_done(false) {
    assert(m_stmt.prepared());
  }
  ~RepoQuery() { m_stmt.reset(); }

  void bindBlob(const char* paramName, const void* blob, size_t size,
                bool isStatic=false);
  void bindBlob(const char* paramName, const BlobEncoder& blob,
                bool isStatic=false);
  void bindMd5(const char* paramName, const MD5& md5);
  void bindTypedValue(const char* paramName, const TypedValue& tv);
  void bindText(const char* paramName, const char* text, size_t size,
                bool isStatic=false);
  void bindStaticString(const char* paramName, const StringData* sd);
  void bindStdString(const char* paramName, const std::string& s);
  void bindStringPiece(const char* paramName, const folly::StringPiece s);
  void bindDouble(const char* paramName, double val);
  void bindInt(const char* paramName, int val);
  void bindId(const char* paramName, Id id);
  void bindOffset(const char* paramName, Offset offset);
  void bindAttr(const char* paramName, Attr attrs);
  void bindBool(const char* paramName, bool b);
  void bindInt64(const char* paramName, int64_t val);
  void bindNull(const char* paramName);

  void step(); // throws(RepoExc)
  bool row() const { return m_row; }
  bool done() const { return m_done; }
  void reset();
  void exec(); // throws(RepoExc)

  // rowid() returns the row id associated with the most recent successful
  // insert.  Thus the rowid is irrelevant for non-insert queries.
  int64_t getInsertedRowid();
  bool isBlob(int iCol);
  bool isText(int iCol);
  bool isDouble(int iCol);
  bool isInt(int iCol);
  bool isNull(int iCol);

  // Get the column value as the named type. If the value cannot be converted
  // into the named type then an error is thrown.
  void getBlob(int iCol, const void*& blob, size_t& size); // throws(RepoExc)
  BlobDecoder getBlob(int iCol); // throws(RepoExc)
  void getMd5(int iCol, MD5& md5); // throws(RepoExc)
  void getTypedValue(int iCol, TypedValue& tv); // throws(RepoExc)
  void getText(int iCol, const char*& text); // throws(RepoExc)
  void getText(int iCol, const char*& text, size_t& size); // throws(RepoExc)
  void getStaticString(int iCol, StringData*& s); // throws(RepoExc)
  void getStdString(int iCol, std::string& s); // throws(RepoExc)
  void getDouble(int iCol, double& val); // throws(RepoExc)
  void getInt(int iCol, int& val); // throws(RepoExc)
  void getId(int iCol, Id& id); // throws(RepoExc)
  void getOffset(int iCol, Offset& offset); // throws(RepoExc)
  void getAttr(int iCol, Attr& attrs); // throws(RepoExc)
  void getBool(int iCol, bool& b); // throws(RepoExc)
  void getInt64(int iCol, int64_t& val); // throws(RepoExc)

 protected:
  RepoStmt& m_stmt;
  bool m_row;
  bool m_done;
};

/*
 * Transaction guard object.
 *
 * Semantics: the guard object will rollback the transaction unless
 * you tell it not to.  Call .commit() when you want things to stay.
 */
struct RepoTxn {
  explicit RepoTxn(Repo& repo); // throws(RepoExc)
  ~RepoTxn();

  RepoTxn(const RepoTxn&) = delete;
  RepoTxn& operator=(const RepoTxn&) = delete;

  /*
   * All these routines may throw if there is an error accessing the
   * repo.  The RepoTxn object will rollback the entire transaction in
   * any of these cases (which technically means they only provide the
   * basic exception safety guarantee, even though the whole point is
   * to behave transactionally. ;)
   */
  void prepare(RepoStmt& stmt, const std::string& sql); // throws(RepoExc)
  void exec(const std::string& sQuery); // throws(RepoExc)
  void commit(); // throws(RepoExc)

  bool error() const { return m_error; }

 private:
  friend struct RepoTxnQuery;
  void step(RepoQuery& query); // throws(RepoExc)
  void exec(RepoQuery& query); // throws(RepoExc)
  void rollback(); // nothrow

  Repo& m_repo;
  bool m_pending;
  bool m_error;

  template<class F> void rollback_guard(const char* func, F f);
};

struct RepoTxnQuery : RepoQuery {
  RepoTxnQuery(RepoTxn& txn, RepoStmt& stmt)
    : RepoQuery(stmt), m_txn(txn) {
  }

  void step(); // throws(RepoExc)
  void exec(); // throws(RepoExc)

 private:
  RepoTxn& m_txn;
};

struct RepoProxy {
  explicit RepoProxy(Repo& repo) : m_repo(repo) {}
  ~RepoProxy() {}

 protected:
  struct Stmt : RepoStmt {
    Stmt(Repo& repo, int repoId) : RepoStmt(repo), m_repoId(repoId) {}
   protected:
    int m_repoId;
  };

  Repo& m_repo;
};

}

#endif