mirror of
https://gitee.com/acl-dev/acl.git
synced 2024-12-14 08:50:53 +08:00
344 lines
8.2 KiB
C
344 lines
8.2 KiB
C
#include "lib_acl.h"
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <assert.h>
|
|
|
|
#ifdef WIN32
|
|
#define snprintf _snprintf
|
|
#endif
|
|
|
|
typedef struct {
|
|
char *buf;
|
|
int len;
|
|
} MYOBJ;
|
|
|
|
static int __debug = 1;
|
|
|
|
static void myobj_free(MYOBJ *o)
|
|
{
|
|
acl_myfree(o->buf);
|
|
acl_myfree(o);
|
|
}
|
|
|
|
static void free_fn(const ACL_CACHE2_INFO *info, void *arg)
|
|
{
|
|
MYOBJ *o = (MYOBJ*) arg;
|
|
|
|
if (__debug) {
|
|
printf("%s: when_timeout: %ld, now: %ld, len: %d, deleted\n",
|
|
info->key, (long) info->when_timeout,
|
|
(long) time(NULL), o->len);
|
|
}
|
|
|
|
myobj_free(o);
|
|
}
|
|
|
|
static MYOBJ *myobj_new(int len)
|
|
{
|
|
MYOBJ *o = (MYOBJ*) acl_mymalloc(sizeof(MYOBJ));
|
|
|
|
o->buf = (char*) acl_mymalloc(len <= 0 ? 100 : len);
|
|
o->len = len;
|
|
return (o);
|
|
}
|
|
|
|
static void walk_fn(ACL_CACHE2_INFO *info, void *arg acl_unused)
|
|
{
|
|
MYOBJ *o = (MYOBJ*) info->value;
|
|
|
|
printf("%s: size: %d; when_timeout: %ld\n",
|
|
info->key, o->len, (long) info->when_timeout);
|
|
}
|
|
|
|
static void test1(int n, int timeout)
|
|
{
|
|
ACL_CACHE2_INFO *info;
|
|
ACL_CACHE2 *cache;
|
|
char key[32];
|
|
MYOBJ *o;
|
|
int i;
|
|
|
|
cache = acl_cache2_create(n, free_fn);
|
|
|
|
for (i = 0; i < n + 5; i++) {
|
|
o = myobj_new(i + 1);
|
|
snprintf(key, sizeof(key), "key(%d)", i);
|
|
assert(acl_cache2_enter(cache, key, o, timeout));
|
|
printf("add one: %s\n", key);
|
|
sleep(1);
|
|
}
|
|
|
|
printf("\nfirst walk cache, cache size: %d\n", acl_cache2_size(cache));
|
|
acl_cache2_walk(cache, walk_fn, NULL);
|
|
printf("\nfirst call acl_cache2_timeout, size: %d\n",
|
|
acl_cache2_size(cache));
|
|
acl_cache2_timeout(cache);
|
|
printf(">>>after first acl_cache2_timeout, second walk cache, "
|
|
"cache's size: %d\n", acl_cache2_size(cache));
|
|
acl_cache2_walk(cache, walk_fn, NULL);
|
|
|
|
printf("\n");
|
|
i = 0;
|
|
while (i++ < 5) {
|
|
printf("slee one second, i=%d\n", i);
|
|
sleep(1);
|
|
}
|
|
|
|
printf("\nsecond call acl_cache_timeout, size: %d\n",
|
|
acl_cache2_size(cache));
|
|
acl_cache2_timeout(cache);
|
|
printf(">>>after second acl_cache_timeout, third walk_cache, cache's "
|
|
"size: %d\n", acl_cache2_size(cache));
|
|
acl_cache2_walk(cache, walk_fn, NULL);
|
|
|
|
o = (MYOBJ*) acl_cache2_find(cache, "key(5)");
|
|
if (o == NULL)
|
|
printf("\n>>>key(5) not exist\n");
|
|
else
|
|
printf("\n>>>key(5): len: %d\n", o->len);
|
|
info = acl_cache2_locate(cache, "key(11)");
|
|
if (info == NULL)
|
|
printf("\n>>>key(11) not exist\n");
|
|
else {
|
|
o = (MYOBJ*) info->value;
|
|
printf("\n>>>key(11): len: %d, when_timeout: %ld\n",
|
|
o->len, (long) info->when_timeout);
|
|
}
|
|
|
|
printf("\nfree cache, size: %d\n", acl_cache2_size(cache));
|
|
printf("\r\n>>>>free all cache nodes(%d)\r\n", __LINE__);
|
|
acl_cache2_free(cache);
|
|
}
|
|
|
|
static void test2(int n, int timeout)
|
|
{
|
|
ACL_CACHE2_INFO *info;
|
|
ACL_CACHE2 *cache;
|
|
char key[32];
|
|
MYOBJ *o;
|
|
int i;
|
|
|
|
cache = acl_cache2_create(n, free_fn);
|
|
|
|
for (i = 0; i < n + 5; i++) {
|
|
o = myobj_new(i + 1);
|
|
snprintf(key, sizeof(key), "key(%d)", i);
|
|
info = (ACL_CACHE2_INFO*) acl_cache2_find(cache, key);
|
|
if (info != NULL)
|
|
printf("the key: %s exist\n", key);
|
|
printf("add one: %s, timeout: %d\n", key, timeout);
|
|
assert(acl_cache2_enter(cache, key, o, timeout));
|
|
}
|
|
|
|
printf(">>>call acl_cache2_walk\n");
|
|
acl_cache2_walk(cache, walk_fn, NULL);
|
|
printf("\r\n>>>>free all cache nodes(%d)\r\n", __LINE__);
|
|
acl_cache2_free(cache);
|
|
}
|
|
|
|
static void test3(int n, int timeout)
|
|
{
|
|
ACL_CACHE2_INFO *info;
|
|
ACL_CACHE2 *cache;
|
|
char key[32];
|
|
MYOBJ *o;
|
|
int i;
|
|
ACL_ITER iter;
|
|
|
|
printf(">>>total add: %d\r\n", n * 2);
|
|
cache = acl_cache2_create(n * 2, free_fn);
|
|
|
|
for (i = 0; i < n; i++) {
|
|
snprintf(key, sizeof(key), "key-%d", 2 * i + 1);
|
|
info = (ACL_CACHE2_INFO*) acl_cache2_find(cache, key);
|
|
if (info != NULL) {
|
|
printf("ALREADY EXIST, key=%s\r\n", key);
|
|
exit (1);
|
|
}
|
|
|
|
o = myobj_new(2 * i + 1);
|
|
if (acl_cache2_enter(cache, key, o, timeout++) == NULL) {
|
|
printf("ADD ERROR, key=%s\r\n", key);
|
|
exit (1);
|
|
} else {
|
|
printf("add ok, key=%s, timeout=%d\r\n", key, timeout);
|
|
}
|
|
|
|
info = (ACL_CACHE2_INFO *) acl_cache2_find(cache, key);
|
|
if (info == NULL) {
|
|
printf("NOT FOUND, key=%s\r\n", key);
|
|
exit (1);
|
|
}
|
|
|
|
o = myobj_new(2 * i + 2);
|
|
snprintf(key, sizeof(key), "key-%d", 2 * i + 2);
|
|
info = (ACL_CACHE2_INFO*) acl_cache2_find(cache, key);
|
|
if (info != NULL) {
|
|
printf("ALREADY EXIST, the key: %s exist\r\n", key);
|
|
exit (1);
|
|
}
|
|
|
|
if (acl_cache2_enter(cache, key, o, timeout) == NULL) {
|
|
printf("ADD ERROR, key=%s\r\n", key);
|
|
exit (1);
|
|
}
|
|
printf("add ok, key=%s, timeout: %d\r\n", key, timeout);
|
|
}
|
|
|
|
printf("\r\n>>>>acl_foreach\r\n");
|
|
acl_foreach(iter, cache) {
|
|
o = (MYOBJ*) iter.data;
|
|
info = (ACL_CACHE2_INFO*) acl_iter_info(iter, cache);
|
|
printf(">>>len=%d, key=%s, when_timeout=%ld\r\n",
|
|
o->len, info->key, (long) info->when_timeout);
|
|
}
|
|
|
|
printf("\r\n>>>>acl_foreach_reverse\r\n");
|
|
acl_foreach_reverse(iter, cache) {
|
|
o = (MYOBJ*) iter.data;
|
|
info = (ACL_CACHE2_INFO*) acl_iter_info(iter, cache);
|
|
printf(">>>len=%d, key=%s, when_timeout=%ld\r\n",
|
|
o->len, info->key, (long) info->when_timeout);
|
|
}
|
|
|
|
|
|
printf("\r\n>>>>free all cache nodes(%d)\r\n", __LINE__);
|
|
acl_cache2_free(cache);
|
|
printf("\r\n>>>>free all cache nodes(%d)\r\n", __LINE__);
|
|
}
|
|
|
|
static void test4_upsert(ACL_CACHE2 *cache, int timeout)
|
|
{
|
|
char key[32];
|
|
MYOBJ *o;
|
|
int i, exist;
|
|
for (i = 0; i < 10; i++) {
|
|
snprintf(key, sizeof(key), "upsert-%d", i);
|
|
o = myobj_new(4096);
|
|
if (!acl_cache2_upsert(cache, key, o, timeout, &exist)) {
|
|
printf("ADD ERROR, key=%s\r\n", key);
|
|
break;
|
|
}
|
|
if (exist)
|
|
myobj_free(o);
|
|
}
|
|
}
|
|
|
|
static void test4(int n, int timeout)
|
|
{
|
|
ACL_CACHE2_INFO *info;
|
|
ACL_CACHE2 *cache = acl_cache2_create(n, free_fn);
|
|
char key[32];
|
|
MYOBJ *o;
|
|
int i;
|
|
|
|
__debug = 0;
|
|
|
|
test4_upsert(cache, timeout);
|
|
printf(">> first current size=%d\r\n", acl_cache2_size(cache));
|
|
test4_upsert(cache, timeout);
|
|
printf(">> second current size=%d\r\n", acl_cache2_size(cache));
|
|
|
|
acl_cache2_walk(cache, walk_fn, NULL);
|
|
|
|
printf("Enter any key to continue ...");
|
|
fflush(stdout);
|
|
getchar();
|
|
|
|
for (i = 0; i < n; i++) {
|
|
snprintf(key, sizeof(key), "key-%d", i);
|
|
info = (ACL_CACHE2_INFO*) acl_cache2_find(cache, key);
|
|
if (info != NULL) {
|
|
printf("ALREADY EXIST, key=%s\r\n", key);
|
|
break;
|
|
}
|
|
|
|
o = myobj_new(4096);
|
|
if (acl_cache2_enter(cache, key, o, timeout) == NULL) {
|
|
printf("ADD ERROR, key=%s\r\n", key);
|
|
break;
|
|
}
|
|
|
|
info = (ACL_CACHE2_INFO*) acl_cache2_find(cache, key);
|
|
if (info == NULL) {
|
|
printf("NOT FOUND, key=%s\r\n", key);
|
|
break;
|
|
}
|
|
|
|
if (i % 1000 == 0) {
|
|
int k = acl_cache2_timeout(cache);
|
|
int size = acl_cache2_size(cache);
|
|
ACL_CACHE2_INFO *head = acl_cache2_head(cache);
|
|
ACL_CACHE2_INFO *tail = acl_cache2_tail(cache);
|
|
printf("del %d, left=%d, head=%s, %ld, tail=%s, %ld\r\n",
|
|
k, size, head ? head->key : "null",
|
|
head->when_timeout, tail ? tail->key : "null",
|
|
tail->when_timeout);
|
|
sleep(1);
|
|
}
|
|
}
|
|
|
|
printf("\r\n>>>>free all cache nodes(%d)\r\n", __LINE__);
|
|
acl_cache2_free(cache);
|
|
}
|
|
|
|
static void usage(const char *procname)
|
|
{
|
|
printf("usage: %s -h [help]\r\n"
|
|
" -n max_size\r\n"
|
|
" -t timeout\r\n"
|
|
" -c cmd[test1|test2|test3|test4, default: test3]\n",
|
|
procname);
|
|
}
|
|
|
|
int main(int argc, char *argv[])
|
|
{
|
|
int n = 10, ch, timeout = 1;
|
|
char cmd[256];
|
|
|
|
ACL_SAFE_STRNCPY(cmd, "test3", sizeof(cmd));
|
|
while ((ch = getopt(argc, argv, "hn:t:a:")) > 0) {
|
|
switch (ch) {
|
|
case 'h':
|
|
usage(argv[0]);
|
|
exit (0);
|
|
case 'n':
|
|
n = atoi(optarg);
|
|
break;
|
|
case 't':
|
|
timeout = atoi(optarg);
|
|
break;
|
|
case 'a':
|
|
ACL_SAFE_STRNCPY(cmd, optarg, sizeof(cmd));
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
#if 0
|
|
(void) acl_mem_slice_init(8, 10240, 100000,
|
|
ACL_SLICE_FLAG_GC2 | ACL_SLICE_FLAG_RTGC_OFF);
|
|
#endif
|
|
if (strcasecmp(cmd, "test1") == 0)
|
|
test1(n, timeout);
|
|
else if (strcasecmp(cmd, "test2") == 0)
|
|
test2(n, timeout);
|
|
else if (strcasecmp(cmd, "test3") == 0)
|
|
test3(n, timeout);
|
|
else if (strcasecmp(cmd, "test4") == 0)
|
|
test4(n, timeout);
|
|
else
|
|
usage(argv[0]);
|
|
|
|
#if 0
|
|
acl_mem_slice_destroy();
|
|
#endif
|
|
|
|
#ifdef ACL_MS_WINDOWS
|
|
printf("Enter any key to exit...\n");
|
|
getchar();
|
|
#endif
|
|
return (0);
|
|
}
|