ViewVC Help
View File | Revision Log | Show Annotations | Revision Graph | Root Listing
root/i-scream/projects/cms/source/idar/idar.c
Revision: 1.1
Committed: Fri Mar 28 15:37:05 2003 UTC (21 years, 8 months ago) by pajs
Content type: text/plain
Branch: MAIN
Log Message:
idar is a "top like" client which displays data collected by ihost.
It currently displays cpu, load, pageing, mem, swap, net and disk io.
There is infastructure to display much more in the future, and to be able
to sort on more than just CPU.

genmergesort is macro written by mtr to sort a linked list.

Inital version 1.0 This compiles and works. It requires libxml2 (but it
may work against libxml, but this has not been checked). It also requires
ukcprog. This will not currently compile on linux (as it lacks strlcpy).

Usage is

idar <-d number_lines_to_display> machine_name port <"fqdn1;fqdn2;fqdn3">

File Contents

# User Rev Content
1 pajs 1.1 #include <stdio.h>
2     #include <string.h>
3     #include <sys/types.h>
4     #include <sys/socket.h>
5     #include <unistd.h>
6     #include <stdlib.h>
7     #include <ukcprog.h>
8    
9     #include <libxml/xmlmemory.h>
10     #include <libxml/parser.h>
11     #include "genmergesort.h"
12    
13     struct host_line_t{
14     char *hostname;
15     int line;
16    
17     struct host_line_t *next;
18     };
19     typedef struct host_line_t host_line_list_t;
20    
21     struct diskio_data_t{
22     char *name;
23    
24     long long read;
25     long long write;
26    
27     struct diskio_data_t *next;
28     };
29     typedef struct diskio_data_t diskio_data_list_t;
30    
31     struct network_data_t{
32     char *name;
33    
34     long long rx;
35     long long tx;
36    
37     struct network_data_t *next;
38     };
39     typedef struct network_data_t network_data_list_t;
40    
41     /*
42     struct disk_data_t{
43     char *name;
44     char *mount_pnt;
45    
46     long long total_space;
47     long long total_used;
48     long long total_avail;
49     // Other data we are less intrested in are not stored
50    
51     struct disk_data_t *next;
52     };
53     typedef struct disk_data_t disk_data_list_t;
54     */
55     #define MAXHOSTNAMESIZE 10
56     struct machine_data_t{
57    
58     char *name;
59    
60     char sysname[MAXHOSTNAMESIZE+1];
61    
62     double cpu_user;
63     double cpu_idle;
64     double cpu_iowait;
65     double cpu_kernel;
66     double cpu_swap;
67     double cpu_used; /* 100 - idle */
68    
69     long long memory_total;
70     long long memory_free;
71     long long memory_used;
72     double memory_used_pecent;
73    
74     long long swap_total;
75     long long swap_free;
76     long long swap_used;
77     double swap_used_pecent;
78    
79     int pages_in;
80     int pages_out;
81    
82     double load_1;
83     double load_5;
84     double load_15;
85    
86     int processes_total;
87     int processes_sleeping;
88     int processes_cpu;
89     int processes_zombie;
90     int processes_stopped;
91    
92     network_data_list_t *network_data_list;
93     long long network_io_total_tx;
94     long long network_io_total_rx;
95    
96     diskio_data_list_t *diskio_data_list;
97     long long disk_io_total_write;
98     long long disk_io_total_read;
99    
100     /* Maybe in the future */
101     /*
102     disk_data_list_t disk_data_list;
103     double disk_total_used;
104     */
105    
106     struct machine_data_t *next;
107     };
108    
109     typedef struct machine_data_t machine_data_list_t;
110    
111     typedef struct{
112     int cpu_user;
113     int cpu_idle;
114     int cpu_iowait;
115     int cpu_kernel;
116     int cpu_swap;
117     int cpu_used;
118    
119     int memory_total;
120     int memory_free;
121     int memory_used;
122     int memory_used_pecent;
123    
124     int swap_total;
125     int swap_free;
126     int swap_used;
127     int swap_used_pecent;
128    
129     int load_1;
130     int load_5;
131     int load_15;
132    
133     int pages_in;
134     int pages_out;
135    
136     int processes_total;
137     int processes_sleeping;
138     int processes_cpu;
139     int processes_zombie;
140     int processes_stopped;
141    
142     int network_io_total_tx;
143     int network_io_total_rx;
144     int network_all_stats;
145    
146     int disk_io_total_write;
147     int disk_io_total_read;
148     int disk_io_all_stats;
149    
150     int disk_total_used;
151     int disk_all_stats;
152     }display_config_t;
153    
154     GENERIC_MERGE_SORT(static, sort_machine_stats, machine_data_list_t, next)
155    
156     int cmp_cpu(machine_data_list_t *a, machine_data_list_t *b){
157    
158     if(a->cpu_used == b->cpu_used){
159     if(a->load_1 == b->load_1) return 0;
160     if(a->load_1 > b->load_1){
161     return -1;
162     }else{
163     return 1;
164     }
165     }
166    
167     if((a->cpu_used) > (b->cpu_used)){
168     return -1;
169     }else{
170     return 1;
171     }
172     }
173    
174     FILE *create_tcp_connection(char *hostname, int port){
175     int sock;
176     struct sockaddr_in addr;
177     struct in_addr haddr;
178     FILE *f;
179    
180     if((sock = socket(PF_INET, SOCK_STREAM, IPPROTO_TCP))<0){
181     return NULL;
182     }
183    
184     if((get_host_addr(hostname, &haddr))!=0){
185     close(sock);
186     return NULL;
187     }
188    
189     memset(&addr, 0, sizeof(addr));
190     addr.sin_family = AF_INET;
191     memcpy(&addr.sin_addr, &haddr, sizeof(haddr));
192     addr.sin_port = htons(port);
193    
194     if(connect(sock, (struct sockaddr *)&addr, sizeof(addr)) !=0){
195     close(sock);
196     return NULL;
197     }
198    
199     if((f=fdopen(sock, "r+"))==NULL){
200     close(sock);
201     return NULL;
202     }
203    
204     return f;
205     }
206    
207     int tcp_comm(FILE *f, char *send, char **response, char *expected){
208    
209     if(send!=NULL){
210     fprintf(f, "%s\n", send);
211     }
212     fflush(f);
213     *response=fpgetline(f);
214     fseek(f, 0, SEEK_CUR);
215    
216     if( (*response==NULL) || (strcmp(*response, "ERROR")==0) ) return -1;
217    
218     if(expected==NULL) return 0;
219    
220     if((strcmp(expected, *response))==0) return 0;
221    
222     return -1;
223     }
224    
225     /* Takes a xml char * and a machine_data_list_t. This will parse
226     * the xml and put it into the correct entry of the machine_data
227     * linked list. This will return the number of entries in the linked
228     * list, or -1 on a error.
229     */
230     int parse_xml(char *xml, machine_data_list_t **md){
231     xmlDocPtr doc;
232     xmlNodePtr cur;
233     xmlNodePtr ele;
234     xmlAttr *list;
235    
236     static int num_hosts=0;
237    
238     machine_data_list_t *machine_data_list=*md;
239    
240     int found_host=0;
241    
242     char *hostname=NULL;
243     network_data_list_t *network_data_list=NULL;
244     network_data_list_t *ndl_ptr=NULL;
245     diskio_data_list_t *diskio_data_list=NULL;
246     diskio_data_list_t *didl_ptr=NULL;
247     char *tmp;
248    
249     doc=xmlParseDoc(xml);
250     if(doc==NULL) return -1;
251    
252     cur = xmlDocGetRootElement(doc);
253     if(cur==NULL){
254     xmlFreeDoc(doc);
255     return -1;
256     }
257    
258    
259     /* Get the hostname */
260     list=cur->properties;
261     while(list!=NULL){
262     if((xmlStrcmp(list->name, (const xmlChar *) "machine_name"))==0){
263     hostname = xmlNodeGetContent(list->children);
264     if(hostname==NULL){
265     return -1;
266     }
267     }
268     list=list->next;
269     }
270     if(hostname==NULL){
271     return -1;
272     }
273    
274     /* Get machine_data for host.. Or create if it doesn't have one */
275     while(machine_data_list!=NULL){
276     if((strncmp(machine_data_list->name, hostname, strlen(hostname)))==0){
277     found_host=1;
278     break;
279     }
280     machine_data_list=machine_data_list->next;
281     }
282    
283     if(!found_host){
284     /* We didn't find this host, but we should be at the end of the list */
285     machine_data_list=malloc(sizeof(machine_data_list_t));
286     if(machine_data_list==NULL) return -1;
287     machine_data_list->next=(*md);
288     machine_data_list->name=hostname;
289     machine_data_list->network_data_list=NULL;
290     machine_data_list->diskio_data_list=NULL;
291     *md=machine_data_list;
292     num_hosts++;
293     }
294    
295     /* Now we want to pull out the data */
296    
297     cur = cur->xmlChildrenNode;
298     while(cur != NULL) {
299     ele=cur->xmlChildrenNode;
300     while(ele != NULL){
301    
302     /* CPU Stats */
303     if(!xmlStrcmp(cur->name, (const xmlChar *) "cpu")){
304     tmp=xmlNodeGetContent(ele);
305     if(!xmlStrcmp(ele->name, (const xmlChar *) "user")){
306     machine_data_list->cpu_user=atof(tmp);
307     }
308     if(!xmlStrcmp(ele->name, (const xmlChar *) "kernel")){
309     machine_data_list->cpu_kernel=atof(tmp);
310     }
311     if(!xmlStrcmp(ele->name, (const xmlChar *) "idle")){
312     machine_data_list->cpu_idle=atof(tmp);
313     }
314     if(!xmlStrcmp(ele->name, (const xmlChar *) "iowait")){
315     machine_data_list->cpu_iowait=atof(tmp);
316     }
317     if(!xmlStrcmp(ele->name, (const xmlChar *) "swap")){
318     machine_data_list->cpu_iowait=atof(tmp);
319     }
320    
321     if(tmp!=NULL) xmlFree(tmp);
322     }
323    
324     /* Memory Stats */
325     if(!xmlStrcmp(cur->name, (const xmlChar *) "memory")){
326     tmp=xmlNodeGetContent(ele);
327     if(!xmlStrcmp(ele->name, (const xmlChar *) "total")){
328     machine_data_list->memory_total=atoll(tmp);
329     }
330     if(!xmlStrcmp(ele->name, (const xmlChar *) "free")){
331     machine_data_list->memory_free=atoll(tmp);
332     }
333     if(!xmlStrcmp(ele->name, (const xmlChar *) "used")){
334     machine_data_list->memory_used=atoll(tmp);
335     }
336    
337     if(tmp!=NULL) xmlFree(tmp);
338     }
339    
340     /* Load Stats */
341     if(!xmlStrcmp(cur->name, (const xmlChar *) "load")){
342     tmp=xmlNodeGetContent(ele);
343     if(!xmlStrcmp(ele->name, (const xmlChar *) "load1")){
344     machine_data_list->load_1=atof(tmp);
345     }
346     if(!xmlStrcmp(ele->name, (const xmlChar *) "load5")){
347     machine_data_list->load_5=atof(tmp);
348     }
349     if(!xmlStrcmp(ele->name, (const xmlChar *) "load15")){
350     machine_data_list->load_15=atof(tmp);
351     }
352    
353     if(tmp!=NULL) xmlFree(tmp);
354     }
355    
356     /* swap stats */
357     if(!xmlStrcmp(cur->name, (const xmlChar *) "swap")){
358     tmp=xmlNodeGetContent(ele);
359     if(!xmlStrcmp(ele->name, (const xmlChar *) "total")){
360     machine_data_list->swap_total=atoll(tmp);
361     }
362     if(!xmlStrcmp(ele->name, (const xmlChar *) "free")){
363     machine_data_list->swap_free=atoll(tmp);
364     }
365     if(!xmlStrcmp(ele->name, (const xmlChar *) "used")){
366     machine_data_list->swap_used=atoll(tmp);
367     }
368    
369     if(tmp!=NULL) xmlFree(tmp);
370     }
371    
372     /* Process stat */
373     if(!xmlStrcmp(cur->name, (const xmlChar *) "processes")){
374     tmp=xmlNodeGetContent(ele);
375     if(!xmlStrcmp(ele->name, (const xmlChar *) "sleeping")){
376     machine_data_list->processes_sleeping=atoi(tmp);
377     }
378     if(!xmlStrcmp(ele->name, (const xmlChar *) "cpu")){
379     machine_data_list->processes_cpu=atoi(tmp);
380     }
381     if(!xmlStrcmp(ele->name, (const xmlChar *) "zombie")){
382     machine_data_list->processes_zombie=atoi(tmp);
383     }
384     if(!xmlStrcmp(ele->name, (const xmlChar *) "total")){
385     machine_data_list->processes_total=atoi(tmp);
386     }
387    
388     if(tmp!=NULL) xmlFree(tmp);
389     }
390    
391     /* paging stats */
392     if(!xmlStrcmp(cur->name, (const xmlChar *) "pages")){
393     tmp=xmlNodeGetContent(ele);
394     if(!xmlStrcmp(ele->name, (const xmlChar *) "pageins")){
395     machine_data_list->pages_in=atoi(tmp);
396     }
397     if(!xmlStrcmp(ele->name, (const xmlChar *) "pageouts")){
398     machine_data_list->pages_out=atoi(tmp);
399     }
400    
401     if(tmp!=NULL) xmlFree(tmp);
402     }
403    
404     /* OS stats */
405     if(!xmlStrcmp(cur->name, (const xmlChar *) "os")){
406     tmp=xmlNodeGetContent(ele);
407     if(!xmlStrcmp(ele->name, (const xmlChar *) "sysname")){
408     strlcpy(machine_data_list->sysname, tmp, MAXHOSTNAMESIZE+1);
409     }
410     if(tmp!=NULL) xmlFree(tmp);
411     }
412    
413     /* Network stats */
414     /* Needs to connect current stat to a previous one. Or create it if new */
415     /* Get name.. Walk list. If match, copy rx/tx values. Else malloc, add to list */
416     if(!xmlStrcmp(cur->name, (const xmlChar *) "net")){
417    
418     list=ele->properties;
419     if(list==NULL) continue;
420     network_data_list=malloc(sizeof(network_data_list_t));
421     if(network_data_list==NULL) return -1;
422     while(list!=NULL){
423     tmp=xmlNodeGetContent(list->children);
424     if(!xmlStrcmp(list->name, (const xmlChar *) "name")){
425     network_data_list->name=strdup(tmp);
426     }
427     if(!xmlStrcmp(list->name, (const xmlChar *) "rx")){
428     network_data_list->rx=atoll(tmp);
429     }
430     if(!xmlStrcmp(list->name, (const xmlChar *) "tx")){
431     network_data_list->tx=atoll(tmp);
432     }
433    
434     xmlFree(tmp);
435     list=list->next;
436     }
437     if(network_data_list->name==NULL) continue;
438     found_host=0;
439     ndl_ptr=machine_data_list->network_data_list;
440     while(ndl_ptr!=NULL){
441     if(!strcmp(ndl_ptr->name, network_data_list->name)){
442     found_host=1;
443     break;
444     }
445     ndl_ptr=ndl_ptr->next;
446     }
447     if(found_host){
448     ndl_ptr->rx=network_data_list->rx;
449     ndl_ptr->tx=network_data_list->tx;
450     free(network_data_list->name);
451     free(network_data_list);
452     }else{
453     network_data_list->next=machine_data_list->network_data_list;
454     machine_data_list->network_data_list=network_data_list;
455     }
456     }
457    
458     /* Disk IO stats */
459     if(!xmlStrcmp(cur->name, (const xmlChar *) "diskio")){
460    
461     list=ele->properties;
462     if(list==NULL) continue;
463     diskio_data_list=malloc(sizeof(diskio_data_list_t));
464     if(diskio_data_list==NULL) return -1;
465     while(list!=NULL){
466     tmp=xmlNodeGetContent(list->children);
467     if(!xmlStrcmp(list->name, (const xmlChar *) "name")){
468     diskio_data_list->name=strdup(tmp);
469     }
470     if(!xmlStrcmp(list->name, (const xmlChar *) "rbytes")){
471     diskio_data_list->read=atoll(tmp);
472     }
473     if(!xmlStrcmp(list->name, (const xmlChar *) "wbytes")){
474     diskio_data_list->write=atoll(tmp);
475     }
476    
477     xmlFree(tmp);
478     list=list->next;
479     }
480     if(diskio_data_list->name==NULL) continue;
481     found_host=0;
482     didl_ptr=machine_data_list->diskio_data_list;
483     while(didl_ptr!=NULL){
484     if(!strcmp(didl_ptr->name, diskio_data_list->name)){
485     found_host=1;
486     break;
487     }
488     didl_ptr=didl_ptr->next;
489     }
490     if(found_host){
491     didl_ptr->read=diskio_data_list->read;
492     didl_ptr->write=diskio_data_list->write;
493     free(diskio_data_list->name);
494     free(diskio_data_list);
495     }else{
496     diskio_data_list->next=machine_data_list->diskio_data_list;
497     machine_data_list->diskio_data_list=diskio_data_list;
498     }
499     }
500    
501    
502    
503     ele=ele->next;
504     }
505     cur=cur->next;
506     }
507    
508     /* Append data we want thats not stored in the server */
509     machine_data_list->cpu_used=100.00-machine_data_list->cpu_idle;
510     machine_data_list->memory_used_pecent=((double)machine_data_list->memory_used / (double)machine_data_list->memory_total) * 100.00;
511     machine_data_list->swap_used_pecent=((double)machine_data_list->swap_used / (double)machine_data_list->swap_total) * 100.00;
512    
513     ndl_ptr=machine_data_list->network_data_list;
514     machine_data_list->network_io_total_tx=0;
515     machine_data_list->network_io_total_rx=0;
516     while(ndl_ptr!=NULL){
517     machine_data_list->network_io_total_tx+=ndl_ptr->tx;
518     machine_data_list->network_io_total_rx+=ndl_ptr->rx;
519     ndl_ptr=ndl_ptr->next;
520     }
521    
522     didl_ptr=machine_data_list->diskio_data_list;
523     machine_data_list->disk_io_total_read=0;
524     machine_data_list->disk_io_total_write=0;
525     while(didl_ptr!=NULL){
526     machine_data_list->disk_io_total_read+=didl_ptr->read;
527     machine_data_list->disk_io_total_write+=didl_ptr->write;
528     didl_ptr=didl_ptr->next;
529     }
530    
531     xmlFreeDoc(doc);
532    
533     return num_hosts;
534    
535    
536     }
537    
538     void display(machine_data_list_t *machine_data_list, int num_lines){
539     int line_num=4;
540    
541     for(;num_lines;num_lines--){
542     if(machine_data_list==NULL) break;
543     printf("\033[%d;%dH%-11s %5.1f %5.1f %5d %5d %5.1f %5.1f %8lld %8lld %9lld %9lld", \
544     line_num++, 1, \
545     machine_data_list->sysname,
546     machine_data_list->cpu_used,
547     machine_data_list->load_1,
548     machine_data_list->pages_in,
549     machine_data_list->pages_out,
550     machine_data_list->memory_used_pecent,
551     machine_data_list->swap_used_pecent,
552     machine_data_list->network_io_total_rx,
553     machine_data_list->network_io_total_tx,
554     machine_data_list->disk_io_total_read,
555     machine_data_list->disk_io_total_write);
556     machine_data_list=machine_data_list->next;
557     }
558    
559     fflush(stdout);
560    
561     }
562    
563     int main(int argc, char **argv){
564     FILE *control;
565     FILE *data;
566    
567     char *machine_list=NULL;
568     char *response=NULL;
569    
570     char *servername;
571     int server_control_port;
572     int server_data_port;
573    
574     machine_data_list_t *machine_data_list=NULL;
575    
576     int num_hosts;
577     int max_display=0;
578    
579     int cmdopt;
580     extern int optind;
581     extern char *optarg;
582    
583     display_config_t display_config;
584    
585     /* What to display defaults */
586     display_config.cpu_user=0;
587     display_config.cpu_idle=0;
588     display_config.cpu_iowait=0;
589     display_config.cpu_kernel=0;
590     display_config.cpu_swap=0;
591     display_config.cpu_used=1;
592    
593     display_config.memory_total=0;
594     display_config.memory_free=0;
595     display_config.memory_used=0;
596     display_config.memory_used_pecent=1;
597    
598     display_config.swap_total=0;
599     display_config.swap_free=0;
600     display_config.swap_used=0;
601     display_config.swap_used_pecent=1;
602    
603     display_config.load_1=1;
604     display_config.load_5=0;
605     display_config.load_15=0;
606    
607     display_config.pages_in=1;
608     display_config.pages_out=1;
609    
610     display_config.processes_total=0;
611     display_config.processes_sleeping=0;
612     display_config.processes_cpu=0;
613     display_config.processes_zombie=0;
614     display_config.processes_stopped=0;
615    
616     display_config.network_io_total_tx=0;
617     display_config.network_io_total_rx=0;
618     display_config.network_all_stats=1;
619    
620     display_config.disk_io_total_write=0;
621     display_config.disk_io_total_read=0;
622     display_config.disk_io_all_stats=1;
623    
624     display_config.disk_total_used=0;
625     display_config.disk_all_stats=0;
626    
627     while((cmdopt=getopt(argc, argv, "d:")) != -1){
628     switch(cmdopt){
629     case 'd':
630     max_display=atoi(optarg);
631     break;
632     }
633     }
634    
635    
636     if(argc<(optind+2)){
637     printf("Usage is %s <-d lines> hostname port <machine list>\n", argv[0]);
638     exit(1);
639     }
640    
641     servername=argv[optind];
642     server_control_port=atoi(argv[optind+1]);
643    
644     control=create_tcp_connection(servername, server_control_port);
645     if(control==NULL){
646     errf("Failed to connect (%m)");
647     }
648    
649     if(argc==4){
650     /* We've been passed a machine list */
651     /* list currently needs to be ; seperated */
652     machine_list=strdup(argv[3]);
653     }
654    
655     if((tcp_comm(control, NULL, &response, "PROTOCOL 1.1"))!=0){
656     errf("Incorrect version number (%s)", response);
657     exit(1);
658     }
659    
660     if((tcp_comm(control, "stattop", &response, "OK"))!=0){
661     errf("Unexpected response %s", response);
662     exit(1);
663     }
664    
665     if(machine_list!=NULL){
666     if((tcp_comm(control, "SETHOSTLIST", &response, "OK"))!=0){
667     errf("Unexpected response %s", response);
668     exit(1);
669     }
670     if((tcp_comm(control, machine_list, &response, "OK"))!=0){
671     errf("Unexpected response %s", response);
672     exit(1);
673     }
674     }
675    
676     if((tcp_comm(control, "STARTDATA", &response, NULL))!=0){
677     errf("Unexpected response %s", response);
678     exit(1);
679     }
680    
681     server_data_port=atoi(response);
682     if(server_data_port==0){
683     errf("Unexpected response %s", response);
684     exit(1);
685     }
686    
687     data=create_tcp_connection(servername, server_data_port);
688     if(data==NULL){
689     errf("Failed to connect to host %s on port %d (%m)",servername, server_data_port);
690     }
691    
692    
693     printf("\033[2J");
694     printf("\033[1;1HHostname CPU Load Page Page Mem Swap Net Net Disk Disk");
695     printf("\033[2;1H used%% (1m) ins outs used used rx tx read write");
696     fflush(stdout);
697     for(;;){
698     response=fpgetline(data);
699     if (response==NULL){
700     errf("Failed to read data (%m)");
701     exit(1);
702     }
703    
704     num_hosts=parse_xml(response, &machine_data_list);
705     if(num_hosts==-1) continue;
706     machine_data_list=sort_machine_stats(machine_data_list, num_hosts, cmp_cpu);
707     if(max_display==0){
708     display(machine_data_list, num_hosts);
709     }else{
710     display(machine_data_list, max_display);
711     }
712    
713     }
714     exit(0);
715     }