Ultrakompakter HTTP Server
ulrich
2024-02-19 63bcdeba5812d7347e2e85302fcf052618e74fcd
commit | author | age
e58690 1 /*
U 2   neon - Embeddable HTTP Server based on jdk.httpserver
3   Copyright (C) 2024  Ulrich Hilger
4
5   This program is free software: you can redistribute it and/or modify
6   it under the terms of the GNU Affero General Public License as
7   published by the Free Software Foundation, either version 3 of the
8   License, or (at your option) any later version.
9
10   This program is distributed in the hope that it will be useful,
11   but WITHOUT ANY WARRANTY; without even the implied warranty of
12   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
13   GNU Affero General Public License for more details.
14
15   You should have received a copy of the GNU Affero General Public License
16   along with this program.  If not, see <https://www.gnu.org/licenses/>.
17  */
18 package de.uhilger.neon;
19
20 import com.sun.net.httpserver.Headers;
21 import com.sun.net.httpserver.HttpExchange;
22 import java.io.File;
23 import java.io.FileInputStream;
24 import java.io.IOException;
25 import java.io.InputStream;
26 import java.io.OutputStream;
27 import java.nio.file.Files;
28 import java.text.SimpleDateFormat;
29 import java.util.Date;
30
31 /**
32  * Helfer zur Beantwortung von HTTP-Anfragen
33  * 
34  * @author Ulrich Hilger
35  * @version 1, 03.06.2021
36  */
37 public class HttpResponder {
38   
39   /* Headernamen */
40   public static final String ACCEPT_RANGES_HEADER = "Accept-Ranges";
41   public static final String CONTENT_LENGTH = "Content-Length";
42   public static final String CONTENT_TYPE = "Content-Type";
43   public static final String LAST_MODIFIED_DATE_HEADER = "Last-Modified";
44
45   /* Statuscodes */
46   public static final int SC_OK = 200;
47   public static final int SC_NOT_FOUND = 404;
48   
49   /* String Konstanten */
50   public static final String STR_BYTES = "bytes";
51   public static final String STR_NOT_FOUND = " not found.";
52   public static final String LM_PATTERN = "EEE, dd MMM yyyy HH:mm:ss zzz";
53   
54   /**
55    * Den Inhalt einer Datei ausliefern
56    *
57    * @param e das Objekt mit Methoden zur Untersuchung der Anfrage sowie zum
58    * Anfertigen und Senden der Antwort
59    * @param file die Datei, deren Inhalt ausgeliefert werden soll
60    * @throws IOException falls etwas schief geht entsteht dieser Fehler
61    */
62   public void serveFile(HttpExchange e, File file) throws IOException {
63     if (file.exists()) {
64       setHeaders(e, file);
65       e.getResponseHeaders().set(CONTENT_LENGTH, Long.toString(file.length()));
66       e.sendResponseHeaders(SC_OK, file.length());
67       if(HttpHelper.HTTP_GET.equalsIgnoreCase(e.getRequestMethod())) {
68         InputStream in = new FileInputStream(file);
69         OutputStream os = e.getResponseBody();        
70         write(in, os);
71         finish(in, os);
72       }
73     } else {
74       sendNotFound(e, file.getName());
75     }
76   }
77
78   public void write(InputStream in, OutputStream out) throws IOException {
79     byte[] b = new byte[4096];
80     int bytesRead = in.read(b);
81     while (bytesRead > -1) {
82       out.write(b, 0, bytesRead);
83       bytesRead = in.read(b);
84     }
85   }
86   
87   public void finish(InputStream in, OutputStream out) throws IOException {
88     in.close();
89     finish(out);
90   }
91
92   public void finish(OutputStream out) throws IOException {
93     out.flush();
94     out.close();
95   }
96   
97   /**
98    * Die Header erzeugen, die unabh&auml;ngig davon, ob der ganze 
99    * Inhalt oder nur Teile davon ausgeliefert werden sollen, in der 
100    * Antwort stehen sollen 
101    * 
102    * @param e das Objekt mit Methoden zur Untersuchung der Anfrage sowie zum
103    * Anfertigen und Senden der Antwort
104    * @param file  die Datei, f&uuml;r die die Header gelten
105    * @throws IOException falls etwas schief geht entsteht dieser Fehler
106    */
107   public void setHeaders(HttpExchange e, File file) throws IOException {
108     Headers resHeaders = e.getResponseHeaders();
109     resHeaders.add(ACCEPT_RANGES_HEADER, STR_BYTES);
110     String mimeType = Files.probeContentType(file.toPath());
111     if (mimeType != null) {
112       resHeaders.add(CONTENT_TYPE, mimeType);
113     }
114     SimpleDateFormat sdf = new SimpleDateFormat(LM_PATTERN);
115     Date date = new Date(file.lastModified());
116     resHeaders.add(LAST_MODIFIED_DATE_HEADER, sdf.format(date));
117   }
118
119   /**
120    * Eine nicht gefunden Antwort senden
121    *
122    * @param e das Objekt mit Methoden zur Untersuchung der Anfrage sowie zum
123    * Anfertigen und Senden der Antwort
124    * @param fname Name der Datei, die nicht gefunden wurde
125    * @throws IOException falls etwas schief geht entsteht dieser Fehler
126    */
127   public void sendNotFound(HttpExchange e, String fname) throws IOException {
128     antwortSenden(e, SC_NOT_FOUND, fname + STR_NOT_FOUND);
129   }  
130
131   public void antwortSenden(HttpExchange exchange, int code, String antwort) throws IOException {
132     byte[] bytes = antwort.getBytes();
133     exchange.sendResponseHeaders(code, bytes.length);
134     OutputStream os = exchange.getResponseBody();
135     os.write(bytes);
136     finish(os);
137   }
138
139
140   
141 }