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 package org.eclipse.jgit.http.server;
45
46 import static javax.servlet.http.HttpServletResponse.SC_NOT_FOUND;
47 import static javax.servlet.http.HttpServletResponse.SC_NOT_MODIFIED;
48 import static org.eclipse.jgit.http.server.ServletUtils.getRepository;
49 import static org.eclipse.jgit.util.HttpSupport.HDR_ETAG;
50 import static org.eclipse.jgit.util.HttpSupport.HDR_IF_MODIFIED_SINCE;
51 import static org.eclipse.jgit.util.HttpSupport.HDR_IF_NONE_MATCH;
52 import static org.eclipse.jgit.util.HttpSupport.HDR_LAST_MODIFIED;
53
54 import java.io.File;
55 import java.io.FileNotFoundException;
56 import java.io.IOException;
57
58 import javax.servlet.ServletException;
59 import javax.servlet.http.HttpServlet;
60 import javax.servlet.http.HttpServletRequest;
61 import javax.servlet.http.HttpServletResponse;
62
63 import org.eclipse.jgit.internal.storage.file.ObjectDirectory;
64 import org.eclipse.jgit.lib.Repository;
65
66
67 abstract class ObjectFileServlet extends HttpServlet {
68 private static final long serialVersionUID = 1L;
69
70 static class Loose extends ObjectFileServlet {
71 private static final long serialVersionUID = 1L;
72
73 Loose() {
74 super("application/x-git-loose-object");
75 }
76
77 @Override
78 String etag(final FileSender sender) throws IOException {
79 return Long.toHexString(sender.getLastModified());
80 }
81 }
82
83 private static abstract class PackData extends ObjectFileServlet {
84 private static final long serialVersionUID = 1L;
85
86 PackData(String contentType) {
87 super(contentType);
88 }
89
90 @Override
91 String etag(final FileSender sender) throws IOException {
92 return sender.getTailChecksum();
93 }
94 }
95
96 static class Pack extends PackData {
97 private static final long serialVersionUID = 1L;
98
99 Pack() {
100 super("application/x-git-packed-objects");
101 }
102 }
103
104 static class PackIdx extends PackData {
105 private static final long serialVersionUID = 1L;
106
107 PackIdx() {
108 super("application/x-git-packed-objects-toc");
109 }
110 }
111
112 private final String contentType;
113
114 ObjectFileServlet(final String contentType) {
115 this.contentType = contentType;
116 }
117
118 abstract String etag(FileSender sender) throws IOException;
119
120 @Override
121 public void doGet(final HttpServletRequest req,
122 final HttpServletResponse rsp) throws IOException {
123 serve(req, rsp, true);
124 }
125
126 @Override
127 protected void doHead(final HttpServletRequest req,
128 final HttpServletResponse rsp) throws ServletException, IOException {
129 serve(req, rsp, false);
130 }
131
132 private void serve(final HttpServletRequest req,
133 final HttpServletResponse rsp, final boolean sendBody)
134 throws IOException {
135 final File obj = new File(objects(req), req.getPathInfo());
136 final FileSender sender;
137 try {
138 sender = new FileSender(obj);
139 } catch (FileNotFoundException e) {
140 rsp.sendError(SC_NOT_FOUND);
141 return;
142 }
143
144 try {
145 final String etag = etag(sender);
146 final long lastModified = (sender.getLastModified() / 1000) * 1000;
147
148 String ifNoneMatch = req.getHeader(HDR_IF_NONE_MATCH);
149 if (etag != null && etag.equals(ifNoneMatch)) {
150 rsp.sendError(SC_NOT_MODIFIED);
151 return;
152 }
153
154 long ifModifiedSince = req.getDateHeader(HDR_IF_MODIFIED_SINCE);
155 if (0 < lastModified && lastModified < ifModifiedSince) {
156 rsp.sendError(SC_NOT_MODIFIED);
157 return;
158 }
159
160 if (etag != null)
161 rsp.setHeader(HDR_ETAG, etag);
162 if (0 < lastModified)
163 rsp.setDateHeader(HDR_LAST_MODIFIED, lastModified);
164 rsp.setContentType(contentType);
165 sender.serve(req, rsp, sendBody);
166 } finally {
167 sender.close();
168 }
169 }
170
171 private static File objects(final HttpServletRequest req) {
172 final Repository db = getRepository(req);
173 return ((ObjectDirectory) db.getObjectDatabase()).getDirectory();
174 }
175 }