Frontend maps request to Atom ID before retrieving resource
[cascardo/atompub.git] / frontend / cgi / cgi.c
1 /*
2  *  Copyright (C) 2007  Thadeu Lima de Souza Cascardo <cascardo@holoscopio.com>
3  *
4  *  This program is free software; you can redistribute it and/or modify
5  *  it under the terms of the GNU General Public License as published by
6  *  the Free Software Foundation; either version 2 of the License, or
7  *  (at your option) any later version.
8  *
9  *  This program is distributed in the hope that it will be useful,
10  *  but WITHOUT ANY WARRANTY; without even the implied warranty of
11  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12  *  GNU General Public License for more details.
13  *
14  *  You should have received a copy of the GNU General Public License along
15  *  with this program; if not, write to the Free Software Foundation, Inc.,
16  *  51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
17  */
18
19
20 #include <atompub/atom.h>
21
22 #include <stdio.h>
23 #include <stdlib.h>
24 #include <string.h>
25 #include <unistd.h>
26
27 void
28 cgi_serve_request (AtomCtx *ctx)
29 {
30   char *method = getenv ("REQUEST_METHOD");
31   char *path = getenv ("PATH_INFO");
32   if (method == NULL)
33     return;
34   if (path == NULL)
35     {
36       if ((path = atom_config_get_str (ctx, "cgi", "index")) == NULL)
37         {
38           /* We do not want to disclose our configuration is empty.
39            * The requester cannot distinguish an empty configuration
40            * from a non-existent index.
41            */
42           fprintf (stdout, "Status: 404 Not Found\n\n");
43           return;
44         }
45     }
46   if (!strcmp (method, "GET"))
47     {
48       AtomID *id;
49       AtomEntry *atom;
50       AtomError *error;
51       char *req;
52       id = atom_id_new_from_frontend (ctx, path);
53       if (id == NULL || (req = atom_id_to_backend (ctx, id)) == NULL)
54         {
55           atom = NULL;
56           error = atom_error_new ();
57           atom_error_code_set (error, 404);
58           atom_error_message_set (error, "Resource not found");
59           atom_error_set (ctx, error);
60         }
61       else
62         {
63           atom = atom_retrieve_entry (ctx, req);
64         }
65       if (id != NULL)
66         atom_id_delete (id);
67       if (atom)
68         {
69           char * str;
70           size_t len;
71           char *header = "Content-type: application/atom+xml\n\n";
72           write (1, header, strlen (header));
73           atom_entry_string (atom, &str, &len);
74           write (1, str, len);
75           g_free (str);
76           atom_entry_delete (atom);
77         }
78       else if ((error = atom_error_get (ctx)) != NULL)
79         {
80           int code = atom_error_code (error);
81           char *message = atom_error_message (error);
82           fprintf (stdout, "Status: %d %s\n\n%s\n", code, message, message);
83         }
84       else
85         {
86           fprintf (stdout, "Status: 500 Server error\n\nServer error\n");
87         }
88     }
89   else
90     {
91       fprintf (stdout, "Status: 501 Not Implemented\n\n");
92     }
93 }
94
95 AtomFrontend *
96 cgi_frontend (void)
97 {
98   AtomFrontend *frontend;
99   frontend = atom_frontend_new ();
100   atom_frontend_map_entries_set (frontend, atom_map_frontend_requests);
101   return frontend;
102 }