Blame t/binary.t

Packit 4e8bc4
#!/usr/bin/perl
Packit 4e8bc4
Packit 4e8bc4
use strict;
Packit 4e8bc4
use warnings;
Packit 4e8bc4
use Test::More;
Packit 4e8bc4
use FindBin qw($Bin);
Packit 4e8bc4
use lib "$Bin/lib";
Packit 4e8bc4
use MemcachedTest;
Packit 4e8bc4
Packit 4e8bc4
my $server = new_memcached("-o no_modern");
Packit 4e8bc4
ok($server, "started the server");
Packit 4e8bc4
Packit 4e8bc4
# Based almost 100% off testClient.py which is:
Packit 4e8bc4
# Copyright (c) 2007  Dustin Sallings <dustin@spy.net>
Packit 4e8bc4
Packit 4e8bc4
# Command constants
Packit 4e8bc4
use constant CMD_GET        => 0x00;
Packit 4e8bc4
use constant CMD_SET        => 0x01;
Packit 4e8bc4
use constant CMD_ADD        => 0x02;
Packit 4e8bc4
use constant CMD_REPLACE    => 0x03;
Packit 4e8bc4
use constant CMD_DELETE     => 0x04;
Packit 4e8bc4
use constant CMD_INCR       => 0x05;
Packit 4e8bc4
use constant CMD_DECR       => 0x06;
Packit 4e8bc4
use constant CMD_QUIT       => 0x07;
Packit 4e8bc4
use constant CMD_FLUSH      => 0x08;
Packit 4e8bc4
use constant CMD_GETQ       => 0x09;
Packit 4e8bc4
use constant CMD_NOOP       => 0x0A;
Packit 4e8bc4
use constant CMD_VERSION    => 0x0B;
Packit 4e8bc4
use constant CMD_GETK       => 0x0C;
Packit 4e8bc4
use constant CMD_GETKQ      => 0x0D;
Packit 4e8bc4
use constant CMD_APPEND     => 0x0E;
Packit 4e8bc4
use constant CMD_PREPEND    => 0x0F;
Packit 4e8bc4
use constant CMD_STAT       => 0x10;
Packit 4e8bc4
use constant CMD_SETQ       => 0x11;
Packit 4e8bc4
use constant CMD_ADDQ       => 0x12;
Packit 4e8bc4
use constant CMD_REPLACEQ   => 0x13;
Packit 4e8bc4
use constant CMD_DELETEQ    => 0x14;
Packit 4e8bc4
use constant CMD_INCREMENTQ => 0x15;
Packit 4e8bc4
use constant CMD_DECREMENTQ => 0x16;
Packit 4e8bc4
use constant CMD_QUITQ      => 0x17;
Packit 4e8bc4
use constant CMD_FLUSHQ     => 0x18;
Packit 4e8bc4
use constant CMD_APPENDQ    => 0x19;
Packit 4e8bc4
use constant CMD_PREPENDQ   => 0x1A;
Packit 4e8bc4
use constant CMD_TOUCH      => 0x1C;
Packit 4e8bc4
use constant CMD_GAT        => 0x1D;
Packit 4e8bc4
use constant CMD_GATQ       => 0x1E;
Packit 4e8bc4
use constant CMD_GATK       => 0x23;
Packit 4e8bc4
use constant CMD_GATKQ      => 0x24;
Packit 4e8bc4
Packit 4e8bc4
# REQ and RES formats are divided even though they currently share
Packit 4e8bc4
# the same format, since they _could_ differ in the future.
Packit 4e8bc4
use constant REQ_PKT_FMT      => "CCnCCnNNNN";
Packit 4e8bc4
use constant RES_PKT_FMT      => "CCnCCnNNNN";
Packit 4e8bc4
use constant INCRDECR_PKT_FMT => "NNNNN";
Packit 4e8bc4
use constant MIN_RECV_BYTES   => length(pack(RES_PKT_FMT));
Packit 4e8bc4
use constant REQ_MAGIC        => 0x80;
Packit 4e8bc4
use constant RES_MAGIC        => 0x81;
Packit 4e8bc4
Packit 4e8bc4
my $mc = MC::Client->new;
Packit 4e8bc4
Packit 4e8bc4
# Let's turn on detail stats for all this stuff
Packit 4e8bc4
Packit 4e8bc4
$mc->stats('detail on');
Packit 4e8bc4
my $check = sub {
Packit 4e8bc4
    my ($key, $orig_flags, $orig_val) = @_;
Packit 4e8bc4
    my ($flags, $val, $cas) = $mc->get($key);
Packit 4e8bc4
    is($flags, $orig_flags, "Flags is set properly");
Packit 4e8bc4
    ok($val eq $orig_val || $val == $orig_val, $val . " = " . $orig_val);
Packit 4e8bc4
};
Packit 4e8bc4
Packit 4e8bc4
my $set = sub {
Packit 4e8bc4
    my ($key, $exp, $orig_flags, $orig_value) = @_;
Packit 4e8bc4
    $mc->set($key, $orig_value, $orig_flags, $exp);
Packit 4e8bc4
    $check->($key, $orig_flags, $orig_value);
Packit 4e8bc4
};
Packit 4e8bc4
Packit 4e8bc4
my $empty = sub {
Packit 4e8bc4
    my $key = shift;
Packit 4e8bc4
    my $rv =()= eval { $mc->get($key) };
Packit 4e8bc4
    is($rv, 0, "Didn't get a result from get");
Packit 4e8bc4
    ok($@->not_found, "We got a not found error when we expected one");
Packit 4e8bc4
};
Packit 4e8bc4
Packit 4e8bc4
my $delete = sub {
Packit 4e8bc4
    my ($key, $when) = @_;
Packit 4e8bc4
    $mc->delete($key, $when);
Packit 4e8bc4
    $empty->($key);
Packit 4e8bc4
};
Packit 4e8bc4
Packit 4e8bc4
# diag "Test Version";
Packit 4e8bc4
my $v = $mc->version;
Packit 4e8bc4
ok(defined $v && length($v), "Proper version: $v");
Packit 4e8bc4
Packit 4e8bc4
# Bug 71
Packit 4e8bc4
{
Packit 4e8bc4
    my %stats1 = $mc->stats('');
Packit 4e8bc4
    $mc->flush;
Packit 4e8bc4
    my %stats2 = $mc->stats('');
Packit 4e8bc4
Packit 4e8bc4
    is($stats2{'cmd_flush'}, $stats1{'cmd_flush'} + 1,
Packit 4e8bc4
       "Stats not updated on a binary flush");
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
# diag "Flushing...";
Packit 4e8bc4
$mc->flush;
Packit 4e8bc4
Packit 4e8bc4
# diag "Noop";
Packit 4e8bc4
$mc->noop;
Packit 4e8bc4
Packit 4e8bc4
# diag "Simple set/get";
Packit 4e8bc4
$set->('x', 5, 19, "somevalue");
Packit 4e8bc4
Packit 4e8bc4
# diag "Delete";
Packit 4e8bc4
$delete->('x');
Packit 4e8bc4
Packit 4e8bc4
# diag "Flush";
Packit 4e8bc4
$set->('x', 5, 19, "somevaluex");
Packit 4e8bc4
$set->('y', 5, 17, "somevaluey");
Packit 4e8bc4
$mc->flush;
Packit 4e8bc4
$empty->('x');
Packit 4e8bc4
$empty->('y');
Packit 4e8bc4
Packit 4e8bc4
{
Packit 4e8bc4
    diag "Some chunked item tests";
Packit 4e8bc4
    my $s2 = new_memcached('-o no_modern,slab_chunk_max=4096');
Packit 4e8bc4
    ok($s2, "started the server");
Packit 4e8bc4
    my $m2 = MC::Client->new($s2);
Packit 4e8bc4
    # Specifically trying to cross the chunk boundary when internally
Packit 4e8bc4
    # appending CLRF.
Packit 4e8bc4
    for my $k (7900..8100) {
Packit 4e8bc4
        my $val = 'd' x $k;
Packit 4e8bc4
        $val .= '123';
Packit 4e8bc4
        $m2->set('t', $val, 0, 0);
Packit 4e8bc4
        # Ensure we get back the same value. Bugs can chop chars.
Packit 4e8bc4
        my (undef, $gval, undef) = $m2->get('t');
Packit 4e8bc4
        ok($gval eq $val, $gval . " = " . $val);
Packit 4e8bc4
    }
Packit 4e8bc4
Packit 4e8bc4
    my $cval = ('d' x 8100) . '123';
Packit 4e8bc4
Packit 4e8bc4
    my $m3 = $s2->new_sock;
Packit 4e8bc4
    mem_get_is($m3, 't', $cval, "large value set from bin fetched from ascii");
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
{
Packit 4e8bc4
    # diag "Add";
Packit 4e8bc4
    $empty->('i');
Packit 4e8bc4
    $mc->add('i', 'ex', 5, 10);
Packit 4e8bc4
    $check->('i', 5, "ex");
Packit 4e8bc4
Packit 4e8bc4
    my $rv =()= eval { $mc->add('i', "ex2", 10, 5) };
Packit 4e8bc4
    is($rv, 0, "Add didn't return anything");
Packit 4e8bc4
    ok($@->exists, "Expected exists error received");
Packit 4e8bc4
    $check->('i', 5, "ex");
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
{
Packit 4e8bc4
    # diag "Too big.";
Packit 4e8bc4
    $empty->('toobig');
Packit 4e8bc4
    $mc->set('toobig', 'not too big', 10, 10);
Packit 4e8bc4
    eval {
Packit 4e8bc4
        my $bigval = ("x" x (1024*1024)) . "x";
Packit 4e8bc4
        $mc->set('toobig', $bigval, 10, 10);
Packit 4e8bc4
    };
Packit 4e8bc4
    ok($@->too_big, "Was too big");
Packit 4e8bc4
    $empty->('toobig');
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
{
Packit 4e8bc4
    # diag "Replace";
Packit 4e8bc4
    $empty->('j');
Packit 4e8bc4
Packit 4e8bc4
    my $rv =()= eval { $mc->replace('j', "ex", 19, 5) };
Packit 4e8bc4
    is($rv, 0, "Replace didn't return anything");
Packit 4e8bc4
    ok($@->not_found, "Expected not_found error received");
Packit 4e8bc4
    $empty->('j');
Packit 4e8bc4
    $mc->add('j', "ex2", 14, 5);
Packit 4e8bc4
    $check->('j', 14, "ex2");
Packit 4e8bc4
    $mc->replace('j', "ex3", 24, 5);
Packit 4e8bc4
    $check->('j', 24, "ex3");
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
{
Packit 4e8bc4
    # diag "MultiGet";
Packit 4e8bc4
    $mc->add('xx', "ex", 1, 5);
Packit 4e8bc4
    $mc->add('wye', "why", 2, 5);
Packit 4e8bc4
    my $rv = $mc->get_multi(qw(xx wye zed));
Packit 4e8bc4
Packit 4e8bc4
    # CAS is returned with all gets.
Packit 4e8bc4
    $rv->{xx}->[2]  = 0;
Packit 4e8bc4
    $rv->{wye}->[2] = 0;
Packit 4e8bc4
    is_deeply($rv->{xx}, [1, 'ex', 0], "X is correct");
Packit 4e8bc4
    is_deeply($rv->{wye}, [2, 'why', 0], "Y is correct");
Packit 4e8bc4
    is(keys(%$rv), 2, "Got only two answers like we expect");
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
# diag "Test increment";
Packit 4e8bc4
$mc->flush;
Packit 4e8bc4
is($mc->incr("x"), 0, "First incr call is zero");
Packit 4e8bc4
is($mc->incr("x"), 1, "Second incr call is one");
Packit 4e8bc4
is($mc->incr("x", 211), 212, "Adding 211 gives you 212");
Packit 4e8bc4
is($mc->incr("x", 2**33), 8589934804, "Blast the 32bit border");
Packit 4e8bc4
Packit 4e8bc4
# diag "Issue 48 - incrementing plain text.";
Packit 4e8bc4
{
Packit 4e8bc4
    $mc->set("issue48", "text", 0, 0);
Packit 4e8bc4
    my $rv =()= eval { $mc->incr('issue48'); };
Packit 4e8bc4
    ok($@ && $@->delta_badval, "Expected invalid value when incrementing text.");
Packit 4e8bc4
    $check->('issue48', 0, "text");
Packit 4e8bc4
Packit 4e8bc4
    $rv =()= eval { $mc->decr('issue48'); };
Packit 4e8bc4
    ok($@ && $@->delta_badval, "Expected invalid value when decrementing text.");
Packit 4e8bc4
    $check->('issue48', 0, "text");
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
# diag "Issue 320 - incr/decr wrong length for initial value";
Packit 4e8bc4
{
Packit 4e8bc4
    $mc->flush;
Packit 4e8bc4
    is($mc->incr("issue320", 1, 1, 0), 1, "incr initial value is 1");
Packit 4e8bc4
    my (undef, $rv, undef) = $mc->get("issue320");
Packit 4e8bc4
    is(length($rv), 1, "initial value length is 1");
Packit 4e8bc4
    is($rv, "1", "initial value is 1");
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
Packit 4e8bc4
# diag "Test decrement";
Packit 4e8bc4
$mc->flush;
Packit 4e8bc4
is($mc->incr("x", undef, 5), 5, "Initial value");
Packit 4e8bc4
is($mc->decr("x"), 4, "Decrease by one");
Packit 4e8bc4
is($mc->decr("x", 211), 0, "Floor is zero");
Packit 4e8bc4
Packit 4e8bc4
{
Packit 4e8bc4
    # diag "bug220
Packit 4e8bc4
    my ($rv, $cas) = $mc->set("bug220", "100", 0, 0);
Packit 4e8bc4
    my ($irv, $icas) = $mc->incr_cas("bug220", 999);
Packit 4e8bc4
    ok($icas != $cas);
Packit 4e8bc4
    is($irv, 1099, "Incr amount failed");
Packit 4e8bc4
    my ($flags, $val, $gcas) = $mc->get("bug220");
Packit 4e8bc4
    is($gcas, $icas, "CAS didn't match after incr/gets");
Packit 4e8bc4
Packit 4e8bc4
    ($irv, $icas) = $mc->incr_cas("bug220", 999);
Packit 4e8bc4
    ok($icas != $cas);
Packit 4e8bc4
    is($irv, 2098, "Incr amount failed");
Packit 4e8bc4
    ($flags, $val, $gcas) = $mc->get("bug220");
Packit 4e8bc4
    is($gcas, $icas, "CAS didn't match after incr/gets");
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
{
Packit 4e8bc4
    # diag "bug21";
Packit 4e8bc4
    $mc->add("bug21", "9223372036854775807", 0, 0);
Packit 4e8bc4
    is($mc->incr("bug21"), 9223372036854775808, "First incr for bug21.");
Packit 4e8bc4
    is($mc->incr("bug21"), 9223372036854775809, "Second incr for bug21.");
Packit 4e8bc4
    is($mc->decr("bug21"), 9223372036854775808, "Decr for bug21.");
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
{
Packit 4e8bc4
    # diag "CAS";
Packit 4e8bc4
    $mc->flush;
Packit 4e8bc4
Packit 4e8bc4
    {
Packit 4e8bc4
        my $rv =()= eval { $mc->set("x", "bad value", 19, 5, 0x7FFFFFF) };
Packit 4e8bc4
        is($rv, 0, "Empty return on expected failure");
Packit 4e8bc4
        ok($@->not_found, "Error was 'not found' as expected");
Packit 4e8bc4
    }
Packit 4e8bc4
Packit 4e8bc4
    my ($r, $rcas) = $mc->add("x", "original value", 5, 19);
Packit 4e8bc4
Packit 4e8bc4
    my ($flags, $val, $i) = $mc->get("x");
Packit 4e8bc4
    is($val, "original value", "->gets returned proper value");
Packit 4e8bc4
    is($rcas, $i, "Add CAS matched.");
Packit 4e8bc4
Packit 4e8bc4
    {
Packit 4e8bc4
        my $rv =()= eval { $mc->set("x", "broken value", 19, 5, $i+1) };
Packit 4e8bc4
        is($rv, 0, "Empty return on expected failure (1)");
Packit 4e8bc4
        ok($@->exists, "Expected error state of 'exists' (1)");
Packit 4e8bc4
    }
Packit 4e8bc4
Packit 4e8bc4
    ($r, $rcas) = $mc->set("x", "new value", 19, 5, $i);
Packit 4e8bc4
Packit 4e8bc4
    my ($newflags, $newval, $newi) = $mc->get("x");
Packit 4e8bc4
    is($newval, "new value", "CAS properly overwrote value");
Packit 4e8bc4
    is($rcas, $newi, "Get CAS matched.");
Packit 4e8bc4
Packit 4e8bc4
    {
Packit 4e8bc4
        my $rv =()= eval { $mc->set("x", "replay value", 19, 5,  $i) };
Packit 4e8bc4
        is($rv, 0, "Empty return on expected failure (2)");
Packit 4e8bc4
        ok($@->exists, "Expected error state of 'exists' (2)");
Packit 4e8bc4
    }
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
# diag "Touch commands";
Packit 4e8bc4
{
Packit 4e8bc4
    $mc->flush;
Packit 4e8bc4
    $mc->set("totouch", "toast", 0, 1);
Packit 4e8bc4
    my $res = $mc->touch("totouch", 10);
Packit 4e8bc4
    sleep 2;
Packit 4e8bc4
    $check->("totouch", 0, "toast");
Packit 4e8bc4
Packit 4e8bc4
    $mc->set("totouch", "toast2", 0, 1);
Packit 4e8bc4
    my ($flags, $val, $i) = $mc->gat("totouch", 10);
Packit 4e8bc4
    is($val, "toast2", "GAT returned correct value");
Packit 4e8bc4
    sleep 2;
Packit 4e8bc4
    $check->("totouch", 0, "toast2");
Packit 4e8bc4
Packit 4e8bc4
    # Test miss as well
Packit 4e8bc4
    $mc->set("totouch", "toast3", 0, 1);
Packit 4e8bc4
    $res = $mc->touch("totouch", 1);
Packit 4e8bc4
    sleep 3;
Packit 4e8bc4
    $empty->("totouch");
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
# diag "Silent set.";
Packit 4e8bc4
$mc->silent_mutation(::CMD_SETQ, 'silentset', 'silentsetval');
Packit 4e8bc4
Packit 4e8bc4
# diag "Silent add.";
Packit 4e8bc4
$mc->silent_mutation(::CMD_ADDQ, 'silentadd', 'silentaddval');
Packit 4e8bc4
Packit 4e8bc4
# diag "Silent replace.";
Packit 4e8bc4
{
Packit 4e8bc4
    my $key = "silentreplace";
Packit 4e8bc4
    my $extra = pack "NN", 829, 0;
Packit 4e8bc4
    $empty->($key);
Packit 4e8bc4
    # $mc->send_silent(::CMD_REPLACEQ, $key, 'somevalue', 7278552, $extra, 0);
Packit 4e8bc4
    # $empty->($key);
Packit 4e8bc4
Packit 4e8bc4
    $mc->add($key, "xval", 831, 0);
Packit 4e8bc4
    $check->($key, 831, 'xval');
Packit 4e8bc4
Packit 4e8bc4
    $mc->send_silent(::CMD_REPLACEQ, $key, 'somevalue', 7278552, $extra, 0);
Packit 4e8bc4
    $check->($key, 829, 'somevalue');
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
# diag "Silent delete";
Packit 4e8bc4
{
Packit 4e8bc4
    my $key = "silentdelete";
Packit 4e8bc4
    $empty->($key);
Packit 4e8bc4
    $mc->set($key, "some val", 19, 0);
Packit 4e8bc4
    $mc->send_silent(::CMD_DELETEQ, $key, '', 772);
Packit 4e8bc4
    $empty->($key);
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
# diag "Silent increment";
Packit 4e8bc4
{
Packit 4e8bc4
    my $key = "silentincr";
Packit 4e8bc4
    my $opaque = 98428747;
Packit 4e8bc4
    $empty->($key);
Packit 4e8bc4
    $mc->silent_incrdecr(::CMD_INCREMENTQ, $key, 0, 0, 0);
Packit 4e8bc4
    is($mc->incr($key, 0), 0, "First call is 0");
Packit 4e8bc4
Packit 4e8bc4
    $mc->silent_incrdecr(::CMD_INCREMENTQ, $key, 8, 0, 0);
Packit 4e8bc4
    is($mc->incr($key, 0), 8);
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
# diag "Silent decrement";
Packit 4e8bc4
{
Packit 4e8bc4
    my $key = "silentdecr";
Packit 4e8bc4
    my $opaque = 98428147;
Packit 4e8bc4
    $empty->($key);
Packit 4e8bc4
    $mc->silent_incrdecr(::CMD_DECREMENTQ, $key, 0, 185, 0);
Packit 4e8bc4
    is($mc->incr($key, 0), 185);
Packit 4e8bc4
Packit 4e8bc4
    $mc->silent_incrdecr(::CMD_DECREMENTQ, $key, 8, 0, 0);
Packit 4e8bc4
    is($mc->incr($key, 0), 177);
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
# diag "Silent flush";
Packit 4e8bc4
{
Packit 4e8bc4
    my %stats1 = $mc->stats('');
Packit 4e8bc4
Packit 4e8bc4
    $set->('x', 5, 19, "somevaluex");
Packit 4e8bc4
    $set->('y', 5, 17, "somevaluey");
Packit 4e8bc4
    $mc->send_silent(::CMD_FLUSHQ, '', '', 2775256);
Packit 4e8bc4
    $empty->('x');
Packit 4e8bc4
    $empty->('y');
Packit 4e8bc4
Packit 4e8bc4
    my %stats2 = $mc->stats('');
Packit 4e8bc4
    is($stats2{'cmd_flush'}, $stats1{'cmd_flush'} + 1,
Packit 4e8bc4
       "Stats not updated on a binary quiet flush");
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
# diag "Append";
Packit 4e8bc4
{
Packit 4e8bc4
    my $key = "appendkey";
Packit 4e8bc4
    my $value = "some value";
Packit 4e8bc4
    $set->($key, 8, 19, $value);
Packit 4e8bc4
    $mc->_append_prepend(::CMD_APPEND, $key, " more");
Packit 4e8bc4
    $check->($key, 19, $value . " more");
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
# diag "Prepend";
Packit 4e8bc4
{
Packit 4e8bc4
    my $key = "prependkey";
Packit 4e8bc4
    my $value = "some value";
Packit 4e8bc4
    $set->($key, 8, 19, $value);
Packit 4e8bc4
    $mc->_append_prepend(::CMD_PREPEND, $key, "prefixed ");
Packit 4e8bc4
    $check->($key, 19, "prefixed " . $value);
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
# diag "Silent append";
Packit 4e8bc4
{
Packit 4e8bc4
    my $key = "appendqkey";
Packit 4e8bc4
    my $value = "some value";
Packit 4e8bc4
    $set->($key, 8, 19, $value);
Packit 4e8bc4
    $mc->send_silent(::CMD_APPENDQ, $key, " more", 7284492);
Packit 4e8bc4
    $check->($key, 19, $value . " more");
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
# diag "Silent prepend";
Packit 4e8bc4
{
Packit 4e8bc4
    my $key = "prependqkey";
Packit 4e8bc4
    my $value = "some value";
Packit 4e8bc4
    $set->($key, 8, 19, $value);
Packit 4e8bc4
    $mc->send_silent(::CMD_PREPENDQ, $key, "prefixed ", 7284492);
Packit 4e8bc4
    $check->($key, 19, "prefixed " . $value);
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
# diag "Leaky binary get test.";
Packit 4e8bc4
# # http://code.google.com/p/memcached/issues/detail?id=16
Packit 4e8bc4
{
Packit 4e8bc4
    # Get a new socket so we can speak text to it.
Packit 4e8bc4
    my $sock = $server->new_sock;
Packit 4e8bc4
    my $max = 1024 * 1024;
Packit 4e8bc4
    my $big = "a big value that's > .5M and < 1M. ";
Packit 4e8bc4
    while (length($big) * 2 < $max) {
Packit 4e8bc4
        $big = $big . $big;
Packit 4e8bc4
    }
Packit 4e8bc4
    my $biglen = length($big);
Packit 4e8bc4
Packit 4e8bc4
    for(1..100) {
Packit 4e8bc4
        my $key = "some_key_$_";
Packit 4e8bc4
        # print STDERR "Key is $key\n";
Packit 4e8bc4
        # print $sock "set $key 0 0 $vallen\r\n$value\r\n";
Packit 4e8bc4
        print $sock "set $key 0 0 $biglen\r\n$big\r\n";
Packit 4e8bc4
        is(scalar <$sock>, "STORED\r\n", "stored big");
Packit 4e8bc4
        my ($f, $v, $c) = $mc->get($key);
Packit 4e8bc4
    }
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
# diag "Test stats settings."
Packit 4e8bc4
{
Packit 4e8bc4
    my %stats = $mc->stats('settings');
Packit 4e8bc4
Packit 4e8bc4
    is(1024, $stats{'maxconns'});
Packit 4e8bc4
    # we run SSL tests over TCP; hence the domain_socket
Packit 4e8bc4
    # is expected to be NULL.
Packit 4e8bc4
    if (enabled_tls_testing()) {
Packit 4e8bc4
        is('NULL', $stats{'domain_socket'});
Packit 4e8bc4
    } else {
Packit 4e8bc4
        isnt('NULL', $stats{'domain_socket'});
Packit 4e8bc4
    }
Packit 4e8bc4
Packit 4e8bc4
    is('on', $stats{'evictions'});
Packit 4e8bc4
    is('yes', $stats{'cas_enabled'});
Packit 4e8bc4
    is('yes', $stats{'flush_enabled'});
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
# diag "Test quit commands.";
Packit 4e8bc4
{
Packit 4e8bc4
    my $s2 = new_memcached();
Packit 4e8bc4
    my $mc2 = MC::Client->new($s2);
Packit 4e8bc4
    $mc2->send_command(CMD_QUITQ, '', '', 0, '', 0);
Packit 4e8bc4
Packit 4e8bc4
    # Five seconds ought to be enough to get hung up on.
Packit 4e8bc4
    my $oldalarmt = alarm(5);
Packit 4e8bc4
Packit 4e8bc4
    # Verify we can't read anything.
Packit 4e8bc4
    my $bytesread = -1;
Packit 4e8bc4
    eval {
Packit 4e8bc4
        local $SIG{'ALRM'} = sub { die "timeout" };
Packit 4e8bc4
        my $data = "";
Packit 4e8bc4
        $bytesread = sysread($mc2->{socket}, $data, 24),
Packit 4e8bc4
    };
Packit 4e8bc4
    is($bytesread, 0, "Read after quit.");
Packit 4e8bc4
Packit 4e8bc4
    # Restore signal stuff.
Packit 4e8bc4
    alarm($oldalarmt);
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
# diag "Test protocol boundary overruns";
Packit 4e8bc4
{
Packit 4e8bc4
    use List::Util qw[min];
Packit 4e8bc4
    # Attempting some protocol overruns by toying around with the edge
Packit 4e8bc4
    # of the data buffer at a few different sizes.  This assumes the
Packit 4e8bc4
    # boundary is at or around 2048 bytes.
Packit 4e8bc4
    for (my $i = 1900; $i < 2100; $i++) {
Packit 4e8bc4
        my $k = "test_key_$i";
Packit 4e8bc4
        my $v = 'x' x $i;
Packit 4e8bc4
        # diag "Trying $i $k";
Packit 4e8bc4
        my $extra = pack "NN", 82, 0;
Packit 4e8bc4
        my $data = $mc->build_command(::CMD_SETQ, $k, $v, 0, $extra, 0);
Packit 4e8bc4
        $data .= $mc->build_command(::CMD_SETQ, "alt_$k", "blah", 0, $extra, 0);
Packit 4e8bc4
        if (length($data) > 2024) {
Packit 4e8bc4
            for (my $j = 2024; $j < min(2096, length($data)); $j++) {
Packit 4e8bc4
                $mc->{socket}->syswrite(substr($data, 0, $j));
Packit 4e8bc4
                $mc->flush_socket;
Packit 4e8bc4
                sleep(0.001);
Packit 4e8bc4
                $mc->{socket}->syswrite(substr($data, $j));
Packit 4e8bc4
                $mc->flush_socket;
Packit 4e8bc4
            }
Packit 4e8bc4
        } else {
Packit 4e8bc4
            $mc->{socket}->syswrite($data);
Packit 4e8bc4
        }
Packit 4e8bc4
        $mc->flush_socket;
Packit 4e8bc4
        $check->($k, 82, $v);
Packit 4e8bc4
        $check->("alt_$k", 82, "blah");
Packit 4e8bc4
    }
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
# Along with the assertion added to the code to verify we're staying
Packit 4e8bc4
# within bounds when we do a stats detail dump (detail turned on at
Packit 4e8bc4
# the top).
Packit 4e8bc4
my %stats = $mc->stats('detail dump');
Packit 4e8bc4
Packit 4e8bc4
# This test causes a disconnection.
Packit 4e8bc4
{
Packit 4e8bc4
    # diag "Key too large.";
Packit 4e8bc4
    my $key = "x" x 365;
Packit 4e8bc4
    eval {
Packit 4e8bc4
        $mc->get($key, 'should die', 10, 10);
Packit 4e8bc4
    };
Packit 4e8bc4
    ok($@->einval, "Invalid key length");
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
done_testing();
Packit 4e8bc4
Packit 4e8bc4
# ######################################################################
Packit 4e8bc4
# Test ends around here.
Packit 4e8bc4
# ######################################################################
Packit 4e8bc4
Packit 4e8bc4
package MC::Client;
Packit 4e8bc4
Packit 4e8bc4
use strict;
Packit 4e8bc4
use warnings;
Packit 4e8bc4
use fields qw(socket);
Packit 4e8bc4
use IO::Socket::INET;
Packit 4e8bc4
Packit 4e8bc4
sub new {
Packit 4e8bc4
    my $self = shift;
Packit 4e8bc4
    my ($s) = @_;
Packit 4e8bc4
    $s = $server unless defined $s;
Packit 4e8bc4
    my $sock = $s->sock;
Packit 4e8bc4
    $self = fields::new($self);
Packit 4e8bc4
    $self->{socket} = $sock;
Packit 4e8bc4
    return $self;
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
sub build_command {
Packit 4e8bc4
    my $self = shift;
Packit 4e8bc4
    die "Not enough args to send_command" unless @_ >= 4;
Packit 4e8bc4
    my ($cmd, $key, $val, $opaque, $extra_header, $cas) = @_;
Packit 4e8bc4
Packit 4e8bc4
    $extra_header = '' unless defined $extra_header;
Packit 4e8bc4
    my $keylen    = length($key);
Packit 4e8bc4
    my $vallen    = length($val);
Packit 4e8bc4
    my $extralen  = length($extra_header);
Packit 4e8bc4
    my $datatype  = 0;  # field for future use
Packit 4e8bc4
    my $reserved  = 0;  # field for future use
Packit 4e8bc4
    my $totallen  = $keylen + $vallen + $extralen;
Packit 4e8bc4
    my $ident_hi  = 0;
Packit 4e8bc4
    my $ident_lo  = 0;
Packit 4e8bc4
Packit 4e8bc4
    if ($cas) {
Packit 4e8bc4
        $ident_hi = int($cas / 2 ** 32);
Packit 4e8bc4
        $ident_lo = int($cas % 2 ** 32);
Packit 4e8bc4
    }
Packit 4e8bc4
Packit 4e8bc4
    my $msg = pack(::REQ_PKT_FMT, ::REQ_MAGIC, $cmd, $keylen, $extralen,
Packit 4e8bc4
                   $datatype, $reserved, $totallen, $opaque, $ident_hi,
Packit 4e8bc4
                   $ident_lo);
Packit 4e8bc4
    my $full_msg = $msg . $extra_header . $key . $val;
Packit 4e8bc4
    return $full_msg;
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
sub send_command {
Packit 4e8bc4
    my $self = shift;
Packit 4e8bc4
    die "Not enough args to send_command" unless @_ >= 4;
Packit 4e8bc4
    my ($cmd, $key, $val, $opaque, $extra_header, $cas) = @_;
Packit 4e8bc4
Packit 4e8bc4
    my $full_msg = $self->build_command($cmd, $key, $val, $opaque, $extra_header, $cas);
Packit 4e8bc4
Packit 4e8bc4
    my $sent = 0;
Packit 4e8bc4
    my $data_len =  length($full_msg);
Packit 4e8bc4
    while ($sent < $data_len) {
Packit 4e8bc4
        my $sent_bytes = $self->{socket}->syswrite($full_msg,
Packit 4e8bc4
                                    $data_len - $sent > MemcachedTest::MAX_READ_WRITE_SIZE ?
Packit 4e8bc4
                                        MemcachedTest::MAX_READ_WRITE_SIZE : ($data_len - $sent),
Packit 4e8bc4
                                    $sent);
Packit 4e8bc4
        last if ($sent_bytes <= 0);
Packit 4e8bc4
        $sent += $sent_bytes;
Packit 4e8bc4
    }
Packit 4e8bc4
    die("Send failed:  $!") unless $data_len;
Packit 4e8bc4
    if($sent != $data_len) {
Packit 4e8bc4
        die("only sent $sent of " . length($full_msg) . " bytes");
Packit 4e8bc4
    }
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
sub flush_socket {
Packit 4e8bc4
    my $self = shift;
Packit 4e8bc4
    $self->{socket}->flush;
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
# Send a silent command and ensure it doesn't respond.
Packit 4e8bc4
sub send_silent {
Packit 4e8bc4
    my $self = shift;
Packit 4e8bc4
    die "Not enough args to send_silent" unless @_ >= 4;
Packit 4e8bc4
    my ($cmd, $key, $val, $opaque, $extra_header, $cas) = @_;
Packit 4e8bc4
Packit 4e8bc4
    $self->send_command($cmd, $key, $val, $opaque, $extra_header, $cas);
Packit 4e8bc4
    $self->send_command(::CMD_NOOP, '', '', $opaque + 1);
Packit 4e8bc4
Packit 4e8bc4
    my ($ropaque, $data) = $self->_handle_single_response;
Packit 4e8bc4
    Test::More::is($ropaque, $opaque + 1);
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
sub silent_mutation {
Packit 4e8bc4
    my $self = shift;
Packit 4e8bc4
    my ($cmd, $key, $value) = @_;
Packit 4e8bc4
Packit 4e8bc4
    $empty->($key);
Packit 4e8bc4
    my $extra = pack "NN", 82, 0;
Packit 4e8bc4
    $mc->send_silent($cmd, $key, $value, 7278552, $extra, 0);
Packit 4e8bc4
    $check->($key, 82, $value);
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
sub _handle_single_response {
Packit 4e8bc4
    my $self = shift;
Packit 4e8bc4
    my $myopaque = shift;
Packit 4e8bc4
Packit 4e8bc4
    my $hdr = "";
Packit 4e8bc4
    while(::MIN_RECV_BYTES - length($hdr) > 0) {
Packit 4e8bc4
        $self->{socket}->sysread(my $response, ::MIN_RECV_BYTES - length($hdr));
Packit 4e8bc4
        $hdr .= $response;
Packit 4e8bc4
    }
Packit 4e8bc4
    Test::More::is(length($hdr), ::MIN_RECV_BYTES, "Expected read length");
Packit 4e8bc4
Packit 4e8bc4
    my ($magic, $cmd, $keylen, $extralen, $datatype, $status, $remaining,
Packit 4e8bc4
        $opaque, $ident_hi, $ident_lo) = unpack(::RES_PKT_FMT, $hdr);
Packit 4e8bc4
    Test::More::is($magic, ::RES_MAGIC, "Got proper response magic");
Packit 4e8bc4
Packit 4e8bc4
    my $cas = ($ident_hi * 2 ** 32) + $ident_lo;
Packit 4e8bc4
Packit 4e8bc4
    return ($opaque, '', $cas, 0) if($remaining == 0);
Packit 4e8bc4
Packit 4e8bc4
    # fetch the value
Packit 4e8bc4
    my $rv="";
Packit 4e8bc4
    while($remaining - length($rv) > 0) {
Packit 4e8bc4
        $self->{socket}->sysread(my $buf, $remaining - length($rv));
Packit 4e8bc4
        $rv .= $buf;
Packit 4e8bc4
    }
Packit 4e8bc4
    if(length($rv) != $remaining) {
Packit 4e8bc4
        my $found = length($rv);
Packit 4e8bc4
        die("Expected $remaining bytes, got $found");
Packit 4e8bc4
    }
Packit 4e8bc4
    if (defined $myopaque) {
Packit 4e8bc4
        Test::More::is($opaque, $myopaque, "Expected opaque");
Packit 4e8bc4
    } else {
Packit 4e8bc4
        Test::More::pass("Implicit pass since myopaque is undefined");
Packit 4e8bc4
    }
Packit 4e8bc4
Packit 4e8bc4
    if ($status) {
Packit 4e8bc4
        die MC::Error->new($status, $rv);
Packit 4e8bc4
    }
Packit 4e8bc4
Packit 4e8bc4
    return ($opaque, $rv, $cas, $keylen);
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
sub _do_command {
Packit 4e8bc4
    my $self = shift;
Packit 4e8bc4
    die unless @_ >= 3;
Packit 4e8bc4
    my ($cmd, $key, $val, $extra_header, $cas) = @_;
Packit 4e8bc4
Packit 4e8bc4
    $extra_header = '' unless defined $extra_header;
Packit 4e8bc4
    my $opaque = int(rand(2**32));
Packit 4e8bc4
    $self->send_command($cmd, $key, $val, $opaque, $extra_header, $cas);
Packit 4e8bc4
    my (undef, $rv, $rcas) = $self->_handle_single_response($opaque);
Packit 4e8bc4
    return ($rv, $rcas);
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
sub _incrdecr_header {
Packit 4e8bc4
    my $self = shift;
Packit 4e8bc4
    my ($amt, $init, $exp) = @_;
Packit 4e8bc4
Packit 4e8bc4
    my $amt_hi = int($amt / 2 ** 32);
Packit 4e8bc4
    my $amt_lo = int($amt % 2 ** 32);
Packit 4e8bc4
Packit 4e8bc4
    my $init_hi = int($init / 2 ** 32);
Packit 4e8bc4
    my $init_lo = int($init % 2 ** 32);
Packit 4e8bc4
Packit 4e8bc4
    my $extra_header = pack(::INCRDECR_PKT_FMT, $amt_hi, $amt_lo, $init_hi,
Packit 4e8bc4
                            $init_lo, $exp);
Packit 4e8bc4
Packit 4e8bc4
    return $extra_header;
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
sub _incrdecr_cas {
Packit 4e8bc4
    my $self = shift;
Packit 4e8bc4
    my ($cmd, $key, $amt, $init, $exp) = @_;
Packit 4e8bc4
Packit 4e8bc4
    my ($data, $rcas) = $self->_do_command($cmd, $key, '',
Packit 4e8bc4
                                           $self->_incrdecr_header($amt, $init, $exp));
Packit 4e8bc4
Packit 4e8bc4
    my $header = substr $data, 0, 8, '';
Packit 4e8bc4
    my ($resp_hi, $resp_lo) = unpack "NN", $header;
Packit 4e8bc4
    my $resp = ($resp_hi * 2 ** 32) + $resp_lo;
Packit 4e8bc4
Packit 4e8bc4
    return $resp, $rcas;
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
sub _incrdecr {
Packit 4e8bc4
    my $self = shift;
Packit 4e8bc4
    my ($v, $c) = $self->_incrdecr_cas(@_);
Packit 4e8bc4
    return $v
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
sub silent_incrdecr {
Packit 4e8bc4
    my $self = shift;
Packit 4e8bc4
    my ($cmd, $key, $amt, $init, $exp) = @_;
Packit 4e8bc4
    my $opaque = 8275753;
Packit 4e8bc4
Packit 4e8bc4
    $mc->send_silent($cmd, $key, '', $opaque,
Packit 4e8bc4
                     $mc->_incrdecr_header($amt, $init, $exp));
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
sub stats {
Packit 4e8bc4
    my $self = shift;
Packit 4e8bc4
    my $key  = shift;
Packit 4e8bc4
    my $cas = 0;
Packit 4e8bc4
    my $opaque = int(rand(2**32));
Packit 4e8bc4
    $self->send_command(::CMD_STAT, $key, '', $opaque, '', $cas);
Packit 4e8bc4
Packit 4e8bc4
    my %rv = ();
Packit 4e8bc4
    my $found_key = '';
Packit 4e8bc4
    my $found_val = '';
Packit 4e8bc4
    do {
Packit 4e8bc4
        my ($op, $data, $cas, $keylen) = $self->_handle_single_response($opaque);
Packit 4e8bc4
        if($keylen > 0) {
Packit 4e8bc4
            $found_key = substr($data, 0, $keylen);
Packit 4e8bc4
            $found_val = substr($data, $keylen);
Packit 4e8bc4
            $rv{$found_key} = $found_val;
Packit 4e8bc4
        } else {
Packit 4e8bc4
            $found_key = '';
Packit 4e8bc4
        }
Packit 4e8bc4
    } while($found_key ne '');
Packit 4e8bc4
    return %rv;
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
sub get {
Packit 4e8bc4
    my $self = shift;
Packit 4e8bc4
    my $key  = shift;
Packit 4e8bc4
    my ($rv, $cas) = $self->_do_command(::CMD_GET, $key, '', '');
Packit 4e8bc4
Packit 4e8bc4
    my $header = substr $rv, 0, 4, '';
Packit 4e8bc4
    my $flags  = unpack("N", $header);
Packit 4e8bc4
Packit 4e8bc4
    return ($flags, $rv, $cas);
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
sub get_multi {
Packit 4e8bc4
    my $self = shift;
Packit 4e8bc4
    my @keys = @_;
Packit 4e8bc4
Packit 4e8bc4
    for (my $i = 0; $i < @keys; $i++) {
Packit 4e8bc4
        $self->send_command(::CMD_GETQ, $keys[$i], '', $i, '', 0);
Packit 4e8bc4
    }
Packit 4e8bc4
Packit 4e8bc4
    my $terminal = @keys + 10;
Packit 4e8bc4
    $self->send_command(::CMD_NOOP, '', '', $terminal);
Packit 4e8bc4
Packit 4e8bc4
    my %return;
Packit 4e8bc4
    while (1) {
Packit 4e8bc4
        my ($opaque, $data) = $self->_handle_single_response;
Packit 4e8bc4
        last if $opaque == $terminal;
Packit 4e8bc4
Packit 4e8bc4
        my $header = substr $data, 0, 4, '';
Packit 4e8bc4
        my $flags  = unpack("N", $header);
Packit 4e8bc4
Packit 4e8bc4
        $return{$keys[$opaque]} = [$flags, $data];
Packit 4e8bc4
    }
Packit 4e8bc4
Packit 4e8bc4
    return %return if wantarray;
Packit 4e8bc4
    return \%return;
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
sub touch {
Packit 4e8bc4
    my $self = shift;
Packit 4e8bc4
    my ($key, $expire) = @_;
Packit 4e8bc4
    my $extra_header = pack "N", $expire;
Packit 4e8bc4
    my $cas = 0;
Packit 4e8bc4
    return $self->_do_command(::CMD_TOUCH, $key, '', $extra_header, $cas);
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
sub gat {
Packit 4e8bc4
    my $self   = shift;
Packit 4e8bc4
    my $key    = shift;
Packit 4e8bc4
    my $expire = shift;
Packit 4e8bc4
    my $extra_header = pack "N", $expire;
Packit 4e8bc4
    my ($rv, $cas) = $self->_do_command(::CMD_GAT, $key, '', $extra_header);
Packit 4e8bc4
Packit 4e8bc4
    my $header = substr $rv, 0, 4, '';
Packit 4e8bc4
    my $flags  = unpack("N", $header);
Packit 4e8bc4
Packit 4e8bc4
    return ($flags, $rv, $cas);
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
sub version {
Packit 4e8bc4
    my $self = shift;
Packit 4e8bc4
    return $self->_do_command(::CMD_VERSION, '', '');
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
sub flush {
Packit 4e8bc4
    my $self = shift;
Packit 4e8bc4
    return $self->_do_command(::CMD_FLUSH, '', '');
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
sub add {
Packit 4e8bc4
    my $self = shift;
Packit 4e8bc4
    my ($key, $val, $flags, $expire) = @_;
Packit 4e8bc4
    my $extra_header = pack "NN", $flags, $expire;
Packit 4e8bc4
    my $cas = 0;
Packit 4e8bc4
    return $self->_do_command(::CMD_ADD, $key, $val, $extra_header, $cas);
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
sub set {
Packit 4e8bc4
    my $self = shift;
Packit 4e8bc4
    my ($key, $val, $flags, $expire, $cas) = @_;
Packit 4e8bc4
    my $extra_header = pack "NN", $flags, $expire;
Packit 4e8bc4
    return $self->_do_command(::CMD_SET, $key, $val, $extra_header, $cas);
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
sub _append_prepend {
Packit 4e8bc4
    my $self = shift;
Packit 4e8bc4
    my ($cmd, $key, $val, $cas) = @_;
Packit 4e8bc4
    return $self->_do_command($cmd, $key, $val, '', $cas);
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
sub replace {
Packit 4e8bc4
    my $self = shift;
Packit 4e8bc4
    my ($key, $val, $flags, $expire) = @_;
Packit 4e8bc4
    my $extra_header = pack "NN", $flags, $expire;
Packit 4e8bc4
    my $cas = 0;
Packit 4e8bc4
    return $self->_do_command(::CMD_REPLACE, $key, $val, $extra_header, $cas);
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
sub delete {
Packit 4e8bc4
    my $self = shift;
Packit 4e8bc4
    my ($key) = @_;
Packit 4e8bc4
    return $self->_do_command(::CMD_DELETE, $key, '');
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
sub incr {
Packit 4e8bc4
    my $self = shift;
Packit 4e8bc4
    my ($key, $amt, $init, $exp) = @_;
Packit 4e8bc4
    $amt = 1 unless defined $amt;
Packit 4e8bc4
    $init = 0 unless defined $init;
Packit 4e8bc4
    $exp = 0 unless defined $exp;
Packit 4e8bc4
Packit 4e8bc4
    return $self->_incrdecr(::CMD_INCR, $key, $amt, $init, $exp);
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
sub incr_cas {
Packit 4e8bc4
    my $self = shift;
Packit 4e8bc4
    my ($key, $amt, $init, $exp) = @_;
Packit 4e8bc4
    $amt = 1 unless defined $amt;
Packit 4e8bc4
    $init = 0 unless defined $init;
Packit 4e8bc4
    $exp = 0 unless defined $exp;
Packit 4e8bc4
Packit 4e8bc4
    return $self->_incrdecr_cas(::CMD_INCR, $key, $amt, $init, $exp);
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
sub decr {
Packit 4e8bc4
    my $self = shift;
Packit 4e8bc4
    my ($key, $amt, $init, $exp) = @_;
Packit 4e8bc4
    $amt = 1 unless defined $amt;
Packit 4e8bc4
    $init = 0 unless defined $init;
Packit 4e8bc4
    $exp = 0 unless defined $exp;
Packit 4e8bc4
Packit 4e8bc4
    return $self->_incrdecr(::CMD_DECR, $key, $amt, $init, $exp);
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
sub noop {
Packit 4e8bc4
    my $self = shift;
Packit 4e8bc4
    return $self->_do_command(::CMD_NOOP, '', '');
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
package MC::Error;
Packit 4e8bc4
Packit 4e8bc4
use strict;
Packit 4e8bc4
use warnings;
Packit 4e8bc4
Packit 4e8bc4
use constant ERR_UNKNOWN_CMD  => 0x81;
Packit 4e8bc4
use constant ERR_NOT_FOUND    => 0x1;
Packit 4e8bc4
use constant ERR_EXISTS       => 0x2;
Packit 4e8bc4
use constant ERR_TOO_BIG      => 0x3;
Packit 4e8bc4
use constant ERR_EINVAL       => 0x4;
Packit 4e8bc4
use constant ERR_NOT_STORED   => 0x5;
Packit 4e8bc4
use constant ERR_DELTA_BADVAL => 0x6;
Packit 4e8bc4
Packit 4e8bc4
use overload '""' => sub {
Packit 4e8bc4
    my $self = shift;
Packit 4e8bc4
    return "Memcache Error ($self->[0]): $self->[1]";
Packit 4e8bc4
};
Packit 4e8bc4
Packit 4e8bc4
sub new {
Packit 4e8bc4
    my $class = shift;
Packit 4e8bc4
    my $error = [@_];
Packit 4e8bc4
    my $self = bless $error, (ref $class || $class);
Packit 4e8bc4
Packit 4e8bc4
    return $self;
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
sub not_found {
Packit 4e8bc4
    my $self = shift;
Packit 4e8bc4
    return $self->[0] == ERR_NOT_FOUND;
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
sub exists {
Packit 4e8bc4
    my $self = shift;
Packit 4e8bc4
    return $self->[0] == ERR_EXISTS;
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
sub too_big {
Packit 4e8bc4
    my $self = shift;
Packit 4e8bc4
    return $self->[0] == ERR_TOO_BIG;
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
sub delta_badval {
Packit 4e8bc4
    my $self = shift;
Packit 4e8bc4
    return $self->[0] == ERR_DELTA_BADVAL;
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
sub einval {
Packit 4e8bc4
    my $self = shift;
Packit 4e8bc4
    return $self->[0] == ERR_EINVAL;
Packit 4e8bc4
}
Packit 4e8bc4
Packit 4e8bc4
# vim: filetype=perl
Packit 4e8bc4