1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
|
#include <string.h>
#include <stdio.h>
#include "net.h"
#include "err.h"
#include "doc.h"
int net_addr(const char *url, struct addr *adr, enum protocol prot_default) {
char *prot_mark;
if ((prot_mark = strstr(url, "://"))) {
static struct {
const char *str;
enum protocol prot;
} prot_str_tbl[] = {
{ "gopher", PROT_GOPHER },
{ "gemini", PROT_GEMINI },
{ "file", PROT_FILE },
};
size_t n = prot_mark - url;
adr->prot = PROT_UNKNOWN;
for (size_t i = 0; i < sizeof prot_str_tbl / sizeof *prot_str_tbl; i++) {
if (!strncmp(prot_str_tbl[i].str, url, n)) {
adr->prot = prot_str_tbl[i].prot;
break;
}
}
url = prot_mark + 3;
} else {
adr->prot = prot_default;
}
if (adr->prot == PROT_FILE) {
adr->host_len = 0;
size_t n = strlen(url);
if (n >= PATH_MAX) return -1;
adr->path_len = n;
memcpy(adr->path, url, n + 1);
} else {
adr->host_len = 0;
while (*url && *url != '/' && adr->host_len < HOST_MAX) {
adr->host[adr->host_len++] = *url++;
}
adr->host[adr->host_len] = 0;
if (*url && *url != '/') {
perr("hostname too long");
return -1;
}
if (*url) url++;
adr->path_len = 0;
while (*url && adr->path_len < HOST_MAX) {
adr->path[adr->path_len++] = *url++;
}
adr->path[adr->path_len] = 0;
if (*url) {
perr("path too long");
return -1;
}
}
return 0;
}
static int file_fetch(const struct addr *adr, struct buf *buf, enum doc_type *doct) {
FILE *f = fopen(adr->path, "r/o");
if (!f) {
perr("file not found");
return -1;
}
buf_init(buf, 1024);
char b[256];
while (fgets(b, sizeof b, f)) {
buf_cat(buf, b, strlen(b));
}
buf_catc(buf, 0);
fclose(f);
*doct = DOC_PLAIN;
return 0;
}
int net_fetch(const struct addr *adr, struct buf *buf, enum doc_type *doct) {
switch (adr->prot) {
case PROT_FILE:
return file_fetch(adr, buf, doct);
default:
perr("unsupported protocol");
return -1;
}
}
|