NAME

    DBIx::QuickORM - Actively maintained Object Relational Mapping that
    makes getting started Quick and has a rich feature set.

EXTREMELY EARLY VERSION WARNING!

    THIS IS A VERY EARLY VERSION!

    About 90% of the functionality from the features section is written.

    About 80% of the features have been listed.

    About 40% of the written code is tested.

    About 10% of the documentation has been written.

    If you want to try it, go for it. Some of the tests give a pretty good
    idea of how to use it.

    DO NOT USE THIS FOR ANYTHING PRODUCTION it is not ready yet.

    The API can and will change!

DESCRIPTION

    An actively maintained ORM tool that is quick and easy to start with,
    but powerful and expandable for long term and larger projects. An
    alternative to DBIx::Class, but not a drop-in replacement.

SCOPE

    The primary scope of this project is to write a good ORM for perl. It
    is very easy to add scope, and try to focus on things outside this
    scope. I am not opposed to such things being written around the ORM
    functionality, afterall the project has a lot of useful code, and
    knowledge of the database. But the primary focus must always be the ORM
    functionality, and it must not suffer in favor of functionality beyond
    that scope.

SYNOPSIS

    FIXME!

MOTIVATION

    The most widely accepted ORM for perl, DBIx::Class is for all intents
    and purposes, dead. There is only 1 maintainer, and that person has
    stated that the project is feature complete. The project will recieve
    no updates apart from critical bugs. The distribution has been marked
    such that it absolutely can never be transferred to anyone else.

    There are 4 ways forward:

    Use DBIx::Class it as it is.

      Many people continue to do this.

    Monkeypatch DBIx::Class

      I know a handful of people who are working on a way to do this that
      is not terrible and will effectively keep DBIx::Class on life
      support.

    Fork DBIx::Class

      I was initially going to take this route. But after a couple hours in
      the codebase I realized I dislike the internals of DBIx::Class almost
      as much as I dislike using its interface.

    Write an alternative

      I decided to take this route. I have never liked DBIx::Class, I find
      it difficult to approach, and it is complicated to start a project
      with it. The interface is unintuitive, and the internals are very
      opaque.

      My goal is to start with the interface, make it approachable, easy to
      start, etc. I also want the interface to be intuitive to use. I also
      want expandability. I also want to make sure I adopt the good ideas
      and capabilities from DBIx::Class. Only a fool would say DBIx::Class
      has nothing of value.

 MAINTENANCE COMMITMENT

    I want to be sure that what happened to DBIx::Class cannot happen to
    this project. I will maintain this as long as I am able. When I am not
    capable I will let others pick up where I left off.

    I am stating here, in the docs, for all to see for all time:

    If I become unable to maintain this project, I approve of others being
    given cpan and github permissions to develop and release this
    distribution.

    Peferably maint will be handed off to someone who has been a
    contributor, or to a group of contributors, If none can be found, or
    none are willing, I trust the cpan toolchain group to takeover.

FEATURE/GOAL OVERVIEW

 Quick to start

    It should be very simple to start a project. The ORM should stay out of
    your way until you want to make it do something for you.

 Intuitive

    Names, interfaces, etc should make sense and be obvious.

 Declarative syntax

    Look at the "DECLARATIVE INTERFACE" section below, or the "SYNOPSIS"
    section above.

 SQL <-> Perl conversion

    It can go either way.

  Generate the perl schema from a populated database.

        my $orm = orm 'MyOrm' => sub {
            # First provide db credentials and connect info
            db { ... };
    
            # Tell DBIx::QuickORM to do the rest
            autofill();
        };
    
        # Built for you by reading from the database.
        my $schema = $orm->schema;

  Generate SQL to populate a database from a schema defined in perl.

    See DBIx::QuickORM::Util::SchemaBuilder for more info.

 Async query support

    Async query support is a key and first class feature of DBIx::QuickORM.

  Single async query - single connection

    Launch an async query on the current connection, then do other stuff
    until it is ready.

    See DBIx::QuickORM::Select::Async for full details. but here are some
    teasers:

        # It can take more args than just \%where, this is just a simply case
        my $async = $orm->async(\%where)->start;
        until ($async->ready) { ... };
        my @rows = $async->all;

    You can also turn any select into an async:

        my $select = $orm->select(...);
        my $async = $orm->async;
        $async->start;

  Multiple concurrent async query support - multiple connections on 1
  process

    DBIx::QuickORM calls this an 'aside'. See DBIx::QuickORM::Select::Aside
    for more detail.

    In this case we have 2 queries executing simultaneously.

        my $aside  = $orm->aside(\%where)->start;    # Runs async query on a new connection
        my $select = $orm->select(\%where);
        my @rows1  = $select->all;
        my @rows2  = $aside->all;

    Note that if both queries return some of the same rows there will only
    be 1 copy in cache, and both @row arrays will have the same object
    reference.

  Multiple concurrent async query support - emulation via forking

    See DBIx::QuickORM::Select::Forked for more detail.

    Similar to the 'aside' functionality above, but instead of running an
    async query on a new connection, a new process is forked, and that
    process does a synchronous query and returns the results. This is
    useful for emulating aside/async with databases that do not support it
    such as SQLite.

 First class inflation and deflation (Conflation)

    Inflation and Deflation of columns is a first-class feature. But since
    saying 'inflation and deflation' every time is a chore DBIx::QuickORM
    shortens the concept to 'conflation'. No, the word "conflation" is not
    actually related to "inflation" or "deflation", but it is an amusing
    pun, specially since it still kind of works with the actual definition
    of "conflation".

    If you specify that a column has a conflator, then using my $val =
    $row->column('name') will give you the inflated form. You can also set
    the column by giving it either the inflated or deflated form. You also
    always have access to the raw values, and asking for either the
    'stored' or 'dirty' value will give the raw form.

    You can also use inflated forms in the %where argument to select/find.

    The rows are also smart enough to check if your inflated forms have
    been mutated and consider the row dirty (in need of saving or
    discarding) after the mutation. This is done by deflating the values to
    compare to the stored form when checking for dirtyness.

    If your inflated values are readonly, locked restricted hashes, or
    objects that implement the 'qorm_immutible' method (and it returns
    true). Then the row is smart enough to skip checking them for mutations
    as they cannot be mutated.

    Oh, also of note, inflated forms do not need to be blessed, nor do they
    even need to be references. You could write a conflator that inflates
    string to have "inflated: " prefixed to them, and no prefix when they
    are raw/deflated. A conflator that encrypts/decrypts passively is also
    possible, assuming the encrypted and decrypted forms are easily
    distinguishable.

  UUID, UUID::Binary, UUID::Stringy

    Automatically inflate and deflate UUID's. Your database can store it as
    a native UUID, a BIN(16), a VARCHAR(36), or whatever. Tell the orm the
    row should be conflated as a UUID and it will just work. You can set
    the value by providing a string, binary data, or anything else the
    conflator recognizes. In the DB it will store the right type, and in
    perl you will get a UUID object.

        schema sub {
            table my_table => sub {
                column thing_uuid => sub {
                    conflate 'UUID'; # OR provide '+Your::Conflator', adds 'DBIx::QuickORM::Conflator::' without the '+'
                };
            };
        };

    DBIx::QuickORM::Conflator::UUID

      Inflates to an object of this class, deflates to whatever the
      database column type is. Object stringifies as a UUID string, and you
      can get both the string and binary value from it through accessors.

      If generating the SQL to populate the db this will tell it the column
      should be the 'UUID' type, and will throw an exception if that type
      is not supported by the db.

    DBIx::QuickORM::Conflator::UUID::Binary

      This is useful only if you are generating the schema SQL to populate
      the db and the db does not support UUID types. This will create the
      column using a binary data type like BIN(16).

    DBIx::QuickORM::Conflator::UUID::Stringy

      This is useful only if you are generating the schema SQL to populate
      the db and the db does not support UUID types. This will create the
      column using a stringy data type like VARCHAR(36).

  JSON, JSON::ASCII

    This conflator will inflate the JSON into a perl data structure and
    deflate it back into a JSON string.

    This uses Cpanel::JSON::XS under the hood.

    DBIx::QuickORM::Conflator::JSON

      Defaults to $json->utf8->encode_json

      This produces a utf8 encoded json string.

    DBIx::QuickORM::Conflator::JSON::ASCII

      Defaults to $json->ascii->encode_json

      This produces an ASCII encoded json string with non-ascii characters
      escaped.

  DateTime - Will not leave a mess with Data::Dumper!

    DBIx::QuickORM::Conflator::DateTime

    This conflator will inflate dates and times into DateTime objects.
    However it also wraps them in an DBIx::QuickORM::Util::Mask object.
    This object hides the DateTime object in a sub { $datetime }. When
    dumped by Data::Dumper you get something like this:

        bless( [
                 '2024-10-26T06:18:45',
                 sub { "DUMMY" }
               ], 'DBIx::QuickORM::Conflator::DateTime' );

    This is much better than spewing the DateTime internals, whcih can take
    several pages of scrollback.

    You can still call any valid DateTime method on this object and it will
    delegate it to the one that is masked beind the coderef.

  Custom conflator

    See the DBIx::QuickORM::Role::Conflator role.

  Custom on the fly

    Declarative:

        my $conflator = conflator NAME => sub {
            inflate { ... };
            deflate { ... };
        };

    OOP:

        my $conflator = DBIx::QuickORM::Conflator->new(
            name => 'NAME',
            inflate => sub { ... },
            defalte => sub { ... }
        );

 Multiple ORM instances for different databases and schemas

        db develop    => sub { ... };
        db staging    => sub { ... };
        db production => sub { ... };
    
        my $app1 = schema app1 => { ... };
        my $app2 = schema app2 { ... };
    
        orm app1_dev => sub {
            db 'develop';
            schema 'app1';
        };
    
        orm app2_prod => sub {
            db 'production';
            schema 'app2';
        };
    
        orm both_stage => sub {
            db 'staging';
    
            # Builds a new schema object, does not modify either original
            schema $app1->merge($app2);
        };

 "Select" object that is very similar to DBIx::Class's ResultSet

    ResultSet was a good idea, regardless of your opinion on DBIx::Class.
    The DBIx::QuickORM::Select objects implement most of the same things.

        my $sel = $orm->select('TABLE/SOURCE', \%where)
        my $sel = $orm->select('TABLE/SOURCE', \%where, $order_by)
        my $sel = $orm->select('TABLE/SOURCE', where => $where, order_by => $order_by, ... );
        $sel = $sel->and(\%where);
        my @rows = $sel->all;
        my $row = $sel->next;
        my $total = $sel->count;

 Find exactly 1 row

        # Throws an exception if multiple rows are found.
        my $row = $orm->find($source, \%where);

 Fetch just the data, no row object (bypasses cache)

        my $data_hashref = $orm->fetch($source, \%where);

 Uses SQL::Abstract under the hood for familiar query syntax

    See SQL::Abstract.

 Built in support for transactions and nested transactions (savepoints)

    See DBIx::QuickORM::Transaction and "TRANSACTIONS" in
    DBIx::QuickORM::ORM for additional details.

    $orm->txn_do(sub { ... });

      Void context will commit if there are no exceptions. It will rollback
      the transaction and re-throw the exception if it encounters one.

    $res = $orm->txn_do(sub { ... });

      Scalar context.

      On success it will commit and return whatever the sub returns, or the
      number 1 if the sub returns nothing, or anything falsy. If you want
      to return a false value you must send it as a ref, or use the list
      context form.

      If an exception is thrown by the block then the transaction will be
      rolled back and $res will be false.

    ($ok, $res_or_err) = $orm->txn_do(sub { ... });

      List context.

      On success it will commit and return (1, $result).

      If an exception occurs in the block then the transaction will be
      rolled back, $ok will be 0, and $ret_or_err will contain the
      exception.

        $orm->txn_do(sub {
            my $txn = shift;
    
            # Nested!
            my ($ok, $res_or_err) = $orm->txn_do(sub { ... });
    
            if ($ok) { $txn->commit }
            else     { $txn->rollback };
    
            # Automatic rollback if an exception is thrown, or if commit is not called
        });
    
        # Commit if no exception is thrown, rollback on exception
        $orm->txn_do(sub { ... });

    Or manually:

        my $txn = $orm->start_txn;
    
        if ($ok) { $txn->commit }
        else     { $txn->rollback };
    
        # Force a rollback unless commit or rollback were called:
        $txn = undef;

 Caching system

    Each DBIx::QuickORM::ORM instance has its own cache object.

  Default cache: Naive, only 1 copy of any row in active memory

    DBIx::QuickORM::Cache::Naive is a basic caching system that insures you
    only have 1 copy of any specific row at any given time (assuming it has
    a primary key, no cahcing is attempted for rows with no primary key).

    Note: If you have multiple ORMs connecting to the same db, they do not
    share a cache and you can end up with the same row in memory twice with
    2 different references.

  'None' cache option to skip caching, every find/select gets a new row
  instance

    You can also choose to use DBIx::QuickORM::Cache::None which is
    basically a no-op for everything meaning there is no cache, every time
    you get an object from the db it is a new copy.

  Write your own cache if you do not like these

    Write your own based on the DBIx::QuickORM::Cache base class.

 Multiple databases supported:

    Database interactions are defined by DBIx::QuickORM::DB subclasses. The
    parent class provides a lot of generic functionality that is fairly
    universal. But the subclasses allow you to specify if a DB does or does
    not support things, how to translate type names from other DBs, etc.

  PostgreSQL

    Tells the ORM what features are supported by PostgreSQL, and how to
    access them.

    See DBIx::QuickORM::DB::PostgreSQL, which uses DBD::Pg under the hood.

  MySQL (Generic)

    Tells the ORM what features are supported by any generic MySQL, and how
    to access them.

    This FULLY supports both DBD::mysql and DBD::MariaDB for connections,
    pick whichever you prefer, the DBIx::QuickORM::DB::MySQL class is aware
    of the differences and will alter behavior accordingly.

  MySQL (Percona)

    Tells the ORM what features are supported by Percona MySQL, and how to
    access them.

    This FULLY supports both DBD::mysql and DBD::MariaDB for connections,
    pick whichever you prefer, the DBIx::QuickORM::DB::MySQL and
    DBIx::QuickORM::DB::Percona classes are aware of the differences and
    will alter behavior accordingly.

  MariaDB

    Tells the ORM what features are supported by MariaDB, and how to access
    them.

    This is essentially MySQL + the extra features MariaDB supports.

    This FULLY supports both DBD::mysql and DBD::MariaDB for connections,
    pick whichever you prefer, the DBIx::QuickORM::DB::MySQL and
    DBIx::QuickORM::DB::MariaDB classes are aware of the differences and
    will alter behavior accordingly.

  SQLite

    Tells the ORM what features are supported by SQLite, and how to access
    them.

    See DBIx::QuickORM::DB::SQLite, which uses DBD::SQLite under the hood.

  Write your own orm <-> db link class

    Take a look at DBIx::QuickORM::DB to see what you need to implement.

 Temporary tables and views

    Each ORM object DBIx::QuickORM::ORM has the static schema it is built
    with, but it also has a second 'connection' schema. Using this second
    schema you can define temporary views and tables (on supported
    databases).

        $orm->create_temp_table(...);
        $orm->create_temp_view(...);

    See the DBIx::QuickORM::ORM documentation for more details.

 Highly functional Row class, ability to use custom ones

    DBIx::QuickORM::Row is the base class for rows, and the default one
    used for rows that are returned. It provides several methods for
    getting/setting columns, including directly accessing stored, pending,
    and inflated values. It also has methods for finding and fetching
    relations.

    This row class does not provide any per-column accessors. For those you
    need one of the following:

    DBIx::QuickORM::Row::AutoAccessors

      This row class uses AUTOLOAD to generate accessors based on column
      names on the fly. So my $val = $row->foo is the same as
      $row->column('foo').

      It also generates accessors for relationships on the fly.

    Create your own row subclasses and tell the schema to use them.

          table foo => sub {
              row_class 'My::Row::Class::Foo';
          };

    Create a class that defines the table and generates a table specific
    row class

      My::Table::Foo.pm:

          package My::Table::Foo
          use DBIx::QuickORM ':TABLE_CLASS';
      
          use DBIx::QuickORM::MetaTable foo => sub {
              column id => ...;
              column foo => ...;
      
              # Declarative keywords are removed after this scope ends.
          };
      
          # There are now accessors for all the columns and relationships.
      
          sub whatever_methods_you_want {
              my $self = shift;
              ...
          }

      Elsware...

          orm MyORM => sub {
              table My::Table::Foo;
      
              # or to load a bunch:
              tables 'My::Table'; # Loads all My::Table::* tables
          };

 Relation mapping and pre-fetching

    TODO: Fill this in.

 Plugin system

    There are a lot of hooks, essentially a plugin is either a codered
    called for all hooks (with params telling you about the hook, or they
    are classes/objects that define the 'qorm_plugin_action()" method or
    that consume the DBIx::QuickORM::Role::Plugin role.

        plugin sub { ... }; # On the fly plugin writing
        plugin Some::Plugin; # Use a plugin class (does not have or need a new method)
        plugin Other::Plugin->new(...); Plugin that needs to be blessed

    Bigger example:

        plugin sub {
            my $self = shift;
            my %params     = @_;
            my $hook       = $params{hook};
            my $return_ref = $params{return_ref};
    
            ...
    
            # if the hook expects you to return a value, instead of modifying a ref
            # in %params, then the return_ref will have a scalar reference to set.
            ${return_ref} = $out if defined($return_ref);
        };

    Define custom plugin hooks in your custom tools:

        plugin_hook NAME => \%params; # Any/All plugins can take action here.

  Current hooks

    auto_conflate => (data_type => $dtype, sql_type => $stype, column =>
    $col, table => $table)

      Use this to automatically inject conflation when auto-generating
      perl-side schema from a populated db.

    post_build => (build_params => \%params, built => $out, built_ref =>
    \$out)

      Called after building an object (ORM, Schema, DB, etc).

    pre_build => (build_params => \%params)

      Called before building an object (ORM, Schema, DB, etc).

    relation_name => (default_name => $alias, table => $table, table_name
    => $tname, fk => $fk)

      use to rename relations when auto-generating perl-side schema from a
      populated db.

    sql_spec => (column => $col, table => $table, sql_spec => $spec)

      Opportunity to modify the DBIx::QuickORM::SQLSpec data for a row.

    sql_spec => (table => $table, sql_spec => sql_spec())

      Opportunity to modify the DBIx::QuickORM::SQLSpec data for a table.

  Ability to customize relationship names when auto-generating perl schema
  from SQL schema

    TODO: Fill this in.

 Does not use Moose under the hood (light weight)

    Most objects in DBIx::QuickORM use Object::HashBase which is what Test2
    uses under the hood. Object::HashBase is very lightweight and
    performant.

    For roles DBIx::QuickORM uses Role::Tiny.

 Using Data::Dumper on a row does not dump all the ORM internals

    DBIx::QuickORM::Row objects need access to the source, and to the orm.
    If a reference to these was simply put into the row objects hashref
    then Data::Dumper is going to work hard to absolutely fill your
    scrollback with useless info every time you dump your row. DBIx::Class
    suffers from this issue.

    For DBIx::QuickORM the source is an DBIx::QuickORM::Source object. And
    it is put into the $row->{source} hash key. But first it is masked
    using DBIx::QuickORM::Util::Mask so that when dumped with Data::Dumper
    you see this:

        bless( {
                 'source' => bless( [
                                      'DBIx::QuickORM::Source=HASH(0x59d72c1c33c8)',
                                      sub { "DUMMY" }
                                    ], 'DBIx::QuickORM::Util::Mask' ),
                 ...
                                  }
               }, 'DBIx::QuickORM::Row' );

    All methods that are valid on DBIx::QuickORM::Source can be called on
    the masked form and they will be delegated to the masked object.

    This + the DateTime conflator mean that rows from DBIx::QuickORM can be
    dumped by Data::Dumper without wiping out your scrollback buffer.

DECLARATIVE INTERFACE

    TODO - Fill this in.

SOURCE

    The source code repository for DBIx-QuickORM can be found at
    http://github.com/exodist/DBIx-QuickORM/.

MAINTAINERS

    Chad Granum <exodist@cpan.org>

AUTHORS

    Chad Granum <exodist@cpan.org>

COPYRIGHT

    Copyright Chad Granum <exodist7@gmail.com>.

    This program is free software; you can redistribute it and/or modify it
    under the same terms as Perl itself.

    See http://dev.perl.org/licenses/