コード例 #1
0
    def test_send_data_idempotence(self, db):
        """We check that reading and writing back in is idempotent.

        This is not obvious because we write to a CSV as intermediate step!
        Special cases, we need to check:
        - missing values
        - dates / (datetimes)
        - (decimals)
        - (64bit integer)
        TODO: Currently we hold hands and pray that these cases actually work!
        """
        df_replace = pd.DataFrame({
            'id': [1],
            'name': ['Chill'],
            'owner': [np.nan],
            'birth': ['2018-03-03']
        })

        reset(db)
        db.send_data(df_replace, 'cats', mode='replace')
        df_in = db.query("SELECT * FROM cats;")
        db.send_data(df_in, 'cats', mode='truncate')
        df_inn = db.query("SELECT * FROM cats;")

        assert (df_in == df_inn).all(axis=None)
コード例 #2
0
ファイル: test_send_query.py プロジェクト: INWTlab/dbrequests
 def test_empty_result_set(self, db):
     """Dealing with empty result sets: #32"""
     reset(db)
     res = db.send_query('select * from cats where id < 0')
     res.columns.values
     assert res.shape == (0, 4)
     assert all(res.columns.values == ['id', 'name', 'owner', 'birth'])
コード例 #3
0
 def test_column_arrangemant_is_maintained(self, db):
     """Insert some data with fliped columns: #24"""
     reset(db)
     df_1 = db.send_query(
         "select birth, name, owner from cats where id = 3;")
     db.send_data(df_1, 'cats', mode='insert')
     df_2 = db.send_query(
         "select birth, name, owner from cats where id = 4;")
     assert (df_1 == df_2).all(axis=None)
コード例 #4
0
    def test_escape_sequences(self, db):
        """Insert some data with escape sequences: #28"""
        reset(db)
        db.send_bulk_query('truncate table cats;')
        db.send_data({'name': ['\\'], 'owner': ['0bnrtZN']}, 'cats')
        res = db.send_query('select name, owner from cats;')

        assert res.name[0] == '\\'
        # all known escape sequences from:
        #   https://dev.mysql.com/doc/refman/8.0/en/load-data.html
        assert res.owner[0] == '0bnrtZN'
コード例 #5
0
    def test_send_data(self, db_connect_args):
        """The connection has local_infile set to 0, so we expect an error."""
        df_add = pd.DataFrame({
            'name': ['Chill'],
            'owner': ['Alex'],
            'birth': ['2018-03-03']
        })

        reset(db_connect_args)
        with pytest.raises((InternalError, OperationalError)):
            db_connect_args.send_data(df_add, 'cats', mode='insert')
コード例 #6
0
    def test_with_system_versioned_table_sync(self, db):
        """
        We check that we can send data using the temporary tables context
        manager, but without temporary tables. These are not implemented for
        system versioned tables in mariadb.
        """
        reset(db)
        res = db.send_query('select id, name, owner from cats',
                            to_pandas=False)
        db.send_data(res, 'hist_cats', 'sync_diffs', with_temp=False)

        # If it doesn't work, we get an error from the database.
        assert True
コード例 #7
0
    def test_delete_happy_path(self, db):
        """First delete all rows, then insert new data"""
        df_replace = pd.DataFrame({
            'id': [1],
            'name': ['Chill'],
            'owner': ['Alex'],
            'birth': ['2018-03-03']
        })

        reset(db)
        db.send_data(df_replace, 'cats', mode='delete')

        df_nrow = db.query("SELECT count(*) as nrows FROM cats;")
        assert df_nrow.nrows[0] == 1
コード例 #8
0
    def test_insert_no_override(self, db):
        """Do not override on duplicate key."""
        df_add = pd.DataFrame({
            'id': [3],
            'name': ['Charlie'],
            'owner': ['River'],
            'birth': ['2016-05-22']
        })

        reset(db)
        db.send_data(df_add, 'cats', mode='insert')
        df_out = db.query("SELECT * FROM cats where id = 3;")

        assert df_out.birth.astype(str)[0] == '2016-05-21'
コード例 #9
0
    def test_insert_happy_path(self, db):
        """Insert some data and check, that it is actually there."""
        df_add = pd.DataFrame({
            'name': ['Chill'],
            'owner': ['Alex'],
            'birth': ['2018-03-03']
        })

        reset(db)
        db.send_data(df_add, 'cats', mode='insert')
        df_out = db.query("select name, owner, birth from cats where id = 4;")

        df_out.birth = df_out.birth.astype(str)
        assert (df_add == df_out).all(axis=None)
コード例 #10
0
    def test_delete_rollback_on_failure(self, db):
        """Delete before insert and check the rollback."""
        df_replace = pd.DataFrame({
            'id': [1],
            'name': ['Chill'],
            'owner': ['Alex'],
            'wrong_col': ['2018-03-03']
        })

        reset(db)
        # InternalError with pymysql
        # OperationalError with mysqldb
        with pytest.raises((OperationalError, InternalError)):
            db.send_data(df_replace, 'cats', mode='delete')

        df_nrow = db.query("SELECT count(*) as nrows FROM cats;")
        assert df_nrow.nrows[0] == 3
コード例 #11
0
    def test_with_system_versioned_table(self, db):
        """
        We check that we can send data using the temporary tables context
        manager, but without temporary tables. These are not implemented for
        system versioned tables in mariadb.
        """
        reset(db)
        res = db.send_query('select id, name, owner from cats',
                            to_pandas=False)
        with pytest.raises((OperationalError, InternalError)):
            db.send_data(res, 'hist_cats', 'update_diffs')
        db.send_data(res, 'hist_cats', 'update_diffs', with_temp=False)
        with pytest.raises((OperationalError, InternalError)):
            db.send_delete(res, 'hist_cats', 'in_join')
        db.send_delete(res, 'hist_cats', 'in_join', with_temp=False)

        # If it doesn't work, we get an error from the database.
        assert True
コード例 #12
0
    def test_send_data_replace(self, db):
        """Send data and replace on duplicate key."""
        df_replace = pd.DataFrame({
            'id': [1],
            'name': ['Chill'],
            'owner': ['Alex'],
            'birth': ['2018-03-03']
        })

        reset(db)
        db.send_data(df_replace, 'cats', mode='replace')

        df_nrow = db.query("SELECT count(*) as nrows FROM cats;")
        assert df_nrow.nrows[0] == 3

        df_out = db.query("SELECT * FROM cats where id = 1;")
        df_out.birth = df_out.birth.astype(str)
        assert (df_replace == df_out).all(axis=None)
コード例 #13
0
    def test_send_data_truncate(self, db):
        """Truncate table before insert."""
        df_replace = pd.DataFrame({
            'id': [1],
            'name': ['Chill'],
            'owner': ['Alex'],
            'birth': ['2018-03-03']
        })

        reset(db)
        db.send_data(df_replace, 'cats', mode='truncate')

        df_nrow = db.query("SELECT count(*) as nrows FROM cats;")
        assert df_nrow.nrows[0] == 1

        df_out = db.query("SELECT * FROM cats;")
        df_out.birth = df_out.birth.astype(str)
        assert (df_replace == df_out).all(axis=None)
コード例 #14
0
    def test_send_data_update(self, db):
        """Check for mode update.

        Update means:
        - we can add rows / new data, similar to insert
        - we can update on duplicate key instead of replace
        - we can update selected columns, maybe just one field + primary key
        """
        df_replace = pd.DataFrame({
            'id': [1, 4],
            'name': ['Chill', 'Pi'],
            'owner': ['Alex', 'Matt'],
            'birth': ['2018-03-03', '2019-08-05']
        })

        reset(db)
        db.send_data(df_replace, 'cats', mode='update')

        # We have a new row:
        df_nrow = db.query("SELECT count(*) as nrows FROM cats;")
        assert df_nrow.nrows[0] == 4

        # Changes are made:
        df_out = db.query("SELECT * FROM cats where id in (1, 4);")
        df_out.birth = df_out.birth.astype(str)
        assert (df_replace == df_out).all(axis=None)

        # We can send partial updates, aka single column
        df_replace_small = pd.DataFrame({
            'id': [2],
            'birth': ['2014-11-13']  # we update this value for id = 2
        })
        df_expected = pd.DataFrame({
            'id': [2],
            'name': ['Cookie'],
            'owner': ['Casey'],
            'birth': ['2014-11-13']
        })
        reset(db)
        db.send_data(df_replace_small, 'cats', mode='update')
        df_out = db.query("SELECT * FROM cats where id = 2;")
        df_out.birth = df_out.birth.astype(str)
        assert (df_expected == df_out).all(axis=None)
コード例 #15
0
 def test_send_query(self, db_connect_args):
     """Test that we have working connection."""
     reset(db_connect_args)
     res = db_connect_args.send_query('select 1 as x')
     assert res.shape == (1, 1)