英英词典server

#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <sqlite3.h>
#include <signal.h>
#include <time.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>


#define  N  16
#define  R  1   //  user register
#define  L  2   //  user login
#define  Q  3   //  query word
#define  H  4   //  history record


#define DATABASE "my.db"
typedef struct 
{
int type;
char name[N];
char data[256];   // password or word
} MSG;


void do_register(int connectfd, MSG *msg, sqlite3 *db)
{
char sqlstr[128];
char *errmsg;

sprintf(sqlstr, "insert into usr values ('%s', '%s')", msg->name, msg->data);
printf("%s\n", sqlstr);
if (sqlite3_exec(db, sqlstr, NULL, NULL, &errmsg) != SQLITE_OK)
{
sqlite3_free(errmsg);
sprintf(msg->data, "user %s already exist!!!", msg->name);
}
else
{
strncpy(msg->data, "OK", 256);
}
send(connectfd, msg, sizeof(MSG), 0);
return;
}
void do_login(int connectfd, MSG *msg, sqlite3 *db)
{
char sqlstr[128];
char *errmsg, **result;
int nrow, ncolumn;
sprintf(sqlstr, "select * from usr where name = '%s' and pass = '%s'", msg->name, msg->data);
if (sqlite3_get_table(db, sqlstr, &result, &nrow, &ncolumn, &errmsg) != SQLITE_OK)
{
printf("error : %s\n", errmsg);
sqlite3_free(errmsg);
}
if (nrow == 0)
{
strncpy(msg->data, "name or password is wrong!!!", 256);
}
else
{
strncpy(msg->data, "OK", 256);
}
send(connectfd, msg, sizeof(MSG), 0);
sqlite3_free_table(result);
return;
}
void get_date(char date[])
{
time_t t;
struct tm *tp;
time(&t);
tp = localtime(&t);
strftime(date, 64, "%Y-%m-%d %H:%M:%S", tp);
return;
}
int do_searchword(int connectfd, MSG *msg)
{
FILE *fp;
char line[300];
char *p;
int len, result;
len = strlen(msg->data);
if ((fp = fopen("dict.txt", "r")) == NULL)
{
strcpy(msg->data, "dict on server can't be opened :(");
send(connectfd, msg, sizeof(MSG), 0);
}
printf("query word is %s\n", msg->data);
while (fgets(line, 300, fp) != NULL)      // fgets 閬囧埌\n鏃跺氨浼氳繑鍥?
{
result = strncmp(msg->data, line, len);
if (result > 0) continue;
if (result < 0 || line[len] != ' ') break;
/*
* 濡傛灉鏌ヨ鐨勫崟璇嶄负ab锛屽綋閬嶅巻鍒癮鏃讹紝result>0//涓嶇‘瀹氭槸鍚︽湁姝ゅ崟璇?
* 褰撳凡缁忛亶鍘嗗埌abc  result<0锛?/鑲畾娌℃湁杩欎釜鍗曡瘝
*/
p = line + len;
while (*p == ' ') p++;
strcpy(msg->data, p);
fclose(fp);
return 1;
}
fclose(fp);


return 0;
}


void do_query(int connectfd, MSG *msg, sqlite3 *db)
{
char sqlstr[128], *errmsg;
int found = 0;
char date[64], word[64];
strcpy(word, msg->data);
found = do_searchword(connectfd, msg);
if ( found )
{
get_date(date);
sprintf(sqlstr, "insert into record values ('%s', '%s', '%s')", msg->name, date, word);
if (sqlite3_exec(db, sqlstr, NULL, NULL, &errmsg) != SQLITE_OK)
{
printf("error : %s\n", errmsg);
sqlite3_free(errmsg);
}
}
else
{
strcpy(msg->data, "not found\n");
}
send(connectfd, msg, sizeof(MSG), 0);
return;
}


int history_callback(void *arg, int f_num, char **f_value, char **f_name)
{
int connectfd;
MSG msg;
connectfd = *(int *)arg;
sprintf(msg.data, "%s : %s", f_value[1], f_value[2]);
send(connectfd, &msg, sizeof(msg), 0);
return 0;
}
void do_history(int connectfd, MSG *msg, sqlite3 *db)
{
char sqlstr[128], *errmsg;


sprintf(sqlstr, "select * from record where name = '%s'", msg->name);
if (sqlite3_exec(db, sqlstr, history_callback, (void *)&connectfd, &errmsg) != SQLITE_OK)
{
printf("error : %s\n", errmsg);
sqlite3_free(errmsg);
}
msg->data[0] = '\0';
send(connectfd, msg, sizeof(MSG), 0);

return;
}
void do_client(int connectfd, sqlite3 *db)
{
MSG msg;
while (recv(connectfd, &msg, sizeof(msg), 0) > 0)  // receive request
{
switch ( msg.type )
{
case R :
do_register(connectfd, &msg, db);
break;
case L :
do_login(connectfd, &msg, db);
break;
case Q :
do_query(connectfd, &msg, db);
break;
case H :
do_history(connectfd, &msg, db);
break;
}
}
printf("client quit\n");
exit(0);
return;
}
int main(int argc, char *argv[])
{
int listenfd, connectfd;
struct sockaddr_in server_addr;
pid_t pid;
sqlite3 *db;
if (argc < 3)
{
printf("Usage : %s <ip> <port>\n", argv[0]);
exit(-1);
}
if (sqlite3_open(DATABASE, &db) != SQLITE_OK)
{
printf("error : %s\n", sqlite3_errmsg(db));
exit(-1);
}
if ((listenfd = socket(PF_INET, SOCK_STREAM, 0)) < 0)
{
perror("fail to socket");
exit(-1);
}
bzero(&server_addr, sizeof(server_addr));
server_addr.sin_family = PF_INET;
server_addr.sin_addr.s_addr = inet_addr(argv[1]);
server_addr.sin_port = htons(atoi(argv[2]));
if (bind(listenfd, (struct sockaddr *)&server_addr, sizeof(server_addr)) < 0)
{
perror("fail to bind");
exit(-1);
}
if (listen(listenfd, 5) < 0)
{
perror("fail to listen");
exit(-1);
}
signal(SIGCHLD, SIG_IGN);// 閬垮厤鍍靛案杩涚▼
while ( 1 )
{
if ((connectfd = accept(listenfd, NULL, NULL)) < 0)
{
perror("fail to accept");
exit(-1);
}
puts("client connect");
if ((pid = fork()) < 0)
{
perror("fail to fork");
exit(-1);
}
if (pid == 0)
{
do_client(connectfd, db);
}
close(connectfd);
}
return 0;
}
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值