您好我正在尝试创建一个我想要这些东西的TCP客户端/服务器。 客户端将给出文件的文件名或文件名路径。 服务器将找到该文件并提供以下详细信息: 权限,大小,所有者,所有者组,修改/创建日期,单词数量,用户ID和优先级,如果出现问题,将这些发送给客户端-1。客户端将打印出详细信息。我做了很多这个东西,但我有一个巨大的问题,所以我不能继续,我的问题是,服务器无法识别文件的路径,但我尝试命名文件和通信它OK。我做错了什么? 提前谢谢
CLIENT
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <netdb.h>
void error(char *msg)
{
perror(msg);
exit(0);
}
int main(int argc, char *argv[])
{
int sockfd, portno, n;
struct sockaddr_in serv_addr;
struct hostent *server;
char buffer[1024];
if (argc < 3) {
fprintf(stderr,"usage %s hostname port\n", argv[0]);
exit(0);
}
portno = atoi(argv[2]);
sockfd = socket(AF_INET, SOCK_STREAM, 0);
if (sockfd < 0)
error("ERROR opening socket");
server = gethostbyname(argv[1]);
if (server == NULL) {
fprintf(stderr,"ERROR, no such host\n");
exit(0);
}
bzero((char *) &serv_addr, sizeof(serv_addr));
serv_addr.sin_family = AF_INET;
bcopy((char *)server->h_addr,
(char *)&serv_addr.sin_addr.s_addr,
server->h_length);
serv_addr.sin_port = htons(portno);
if (connect(sockfd,&serv_addr,sizeof(serv_addr)) < 0)
error("ERROR connecting");
printf("Please enter the filename or path of filename: ");
bzero(buffer,1024);
fgets(buffer,1024,stdin);
n = write(sockfd,buffer,strlen(buffer));
if (n < 0)
error("ERROR writing to socket");
bzero(buffer,1024);
n = read(sockfd,buffer,1024);
if (n < 0)
error("ERROR reading from socket");
printf("%s\n",buffer);
return 0;
SERVER
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <sys/resource.h>
#include <sys/time.h>
void error(char *msg)
{
perror(msg);
exit(1);
}
int main(int argc, char *argv[])
{
FILE *fp;
int sockfd, newsockfd, portno, clilen,i;
char buffer[1024],filename[1024];
char * pPath;
struct sockaddr_in serv_addr, cli_addr;
int n;
int which = PRIO_PROCESS;
id_t pid;
int ret;
if (argc < 2) {
fprintf(stderr,"ERROR, no port provided\n");
exit(1);
}
sockfd = socket(AF_INET, SOCK_STREAM, 0);
if (sockfd < 0)
error("ERROR opening socket");
bzero((char *) &serv_addr, sizeof(serv_addr));
portno = atoi(argv[1]);
serv_addr.sin_family = AF_INET;
serv_addr.sin_addr.s_addr = INADDR_ANY;
serv_addr.sin_port = htons(portno);
if (bind(sockfd, (struct sockaddr *) &serv_addr,
sizeof(serv_addr)) < 0)
error("ERROR on binding");
listen(sockfd,5);
clilen = sizeof(cli_addr);
newsockfd = accept(sockfd,
(struct sockaddr *) &cli_addr,
&clilen);
if (newsockfd < 0)
error("ERROR on accept");
bzero(buffer,1024);
n = read(newsockfd,buffer,1024);
if (n < 0) error("ERROR reading from socket");
printf("Here is the message: %s\n",buffer);
system("ls -al 1.txt > ls.txt");
system("wc -w 1.txt > wc.txt");
/* pPath = getenv ("PATH");
if (pPath!=NULL)
printf ("The current path is: %s\n",pPath);
system("touch path.txt");
fp=fopen("path.txt","w");
if (fp==NULL) exit(1);
fprintf(fp,pPath);
fclose(fp); */
pid = getpid();
ret = getpriority(which, pid);
printf("priority %d user id %d ret %d",which,pid,ret);
system("paste ls.txt wc.txt user.txt > info.txt");
fp=fopen("info.txt","r");
if (fp==NULL) exit(1);
for(int i=0;i<1000;i++){
fscanf(fp,"%c",&buffer[i]);
}
fclose(fp);
n = write(newsockfd,buffer,1024);
if (n < 0) error("ERROR writing to socket");
return 0;
}
}`
谢谢你的快速回复。具体来说,教授没有要求我们制作我创建的这个txt文件,但是我创建了因为我找不到解决方案而且我无法将所有内容写入缓冲区。它应该是这样的:
缓冲区我来自客户端的路径 从缓冲区找到路径文件(这是我有问题的地方) ls -al写入缓冲区(我把它写入.txt) wc写入缓冲区(我把它写入.txt) path写入缓冲区(尚未完成此操作) 用户信息写入缓冲区(尚未完成此操作) 缓冲区发送给客户端 客户端打印带有所有这些信息的缓冲区
我尝试了sprintf,但我并不完全明白你是如何使用它的,但是这个命令看起来比我好了谢谢:)。如果我理解的话,没有通过缓冲区。
答案 0 :(得分:0)
您可能需要将路径传递给系统调用。分配一个缓冲区(请记住,您应该对缓冲区进行检查以确保安全,但我知道这可能是功课)并使用sprintf(http://www.cplusplus.com/reference/clibrary/cstdio/sprintf/) :
char lsbuf[1024];
sprintf(lsbuf,"ls -al %s > ls.txt",buffer);
system(lsbuf);
正确的做事方式(以及教师最有可能打算这样做的方式)是在代码中完成ls,wc等的工作。像这样的呼叫系统让你可以接受全新的安全漏洞。
答案 1 :(得分:0)
传递给服务器的路径是否包含尾随'\ n'?如果是这样,你应该删除它(例如通过放置\ 0字符)
答案 2 :(得分:0)
在您当前的代码中,客户输入的内容并没有真正完成,并且行为与您的描述不符。
这应该做什么?:
for(int i=0;i<1000;i++){
fscanf(fp,"%c",&buffer[i]);
}
fclose(fp);
这将查找输入中的每个字符(看到你在这里使用1000,缓冲区可以是1024个字符,你缺少最后24个字符)并返回它在“info.txt”中出现的次数。请注意,您还将搜索NUL字符('\ 0')。除了这些粗心大意之外,它并没有真正做任何事情。
我假设您只是想在文件中找到字符串,并使用行号?您需要一种不同的搜索字符串的方法。有一些聪明且不那么聪明的算法。其中一个更“粗暴”的算法是:
请注意,算法比这更有效。
我没有仔细检查您的网络设置,假设它有效。最好是使用更多的空格将代码分成几个段落,或者将网络设置放入不同的功能中。
答案 3 :(得分:0)
首先,应该避免使用bzero,因为该函数已被弃用。请改用memset。
函数fgets,读取直到到达换行符或文件结尾。由于换行符是有效字符,因此将其添加到字符串中并附加空字符。
如果输入“mytext”然后输入,缓冲区将显示“mytext \ n \ 0”。
根据当前代码,你甚至没有使用来自客户端的缓冲区,因此很明显它不起作用(不是由于附加换行符导致的fgets()行为)。
答案 4 :(得分:0)
很抱歉这样回复,但我没有帐户,我是新手(我的项目没有问题4年:P)。我插入
char lsbuf[1024];
sprintf(lsbuf,"ls -al %s > ls.txt",buffer);
system(lsbuf);
并且几乎像魅力一样...它接受路径并做ls但不保存ls到ls.txt(创建空文件)并且不将它发送到缓冲区。也是我尝试更换memset并正在工作!!