bes Updated for version 3.21.1
The Backend Server (BES) is the lower two tiers of the Hyrax data server
JPEG2000Transmitter.cc
1// JPEG2000Transmitter.cc
2
3// This file is part of BES GDAL File Out Module
4
5// Copyright (c) 2012 OPeNDAP, Inc.
6// Author: James Gallagher <jgallagher@opendap.org>
7//
8// This library is free software; you can redistribute it and/or
9// modify it under the terms of the GNU Lesser General Public
10// License as published by the Free Software Foundation; either
11// version 2.1 of the License, or (at your option) any later version.
12//
13// This library is distributed in the hope that it will be useful,
14// but WITHOUT ANY WARRANTY; without even the implied warranty of
15// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
16// Lesser General Public License for more details.
17//
18// You should have received a copy of the GNU Lesser General Public
19// License along with this library; if not, write to the Free Software
20// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
21//
22// You can contact University Corporation for Atmospheric Research at
23// 3080 Center Green Drive, Boulder, CO 80301
24
25#include "config.h"
26
27#include <unistd.h>
28
29#include <cstdio>
30#include <cstdlib>
31
32#include <sys/types.h> // For umask
33#include <sys/stat.h>
34
35#include <iostream>
36#include <fstream>
37
38#include <libdap/DataDDS.h>
39#include <libdap/BaseType.h>
40#include <libdap/escaping.h>
41
42using namespace libdap;
43
44#include "JPEG2000Transmitter.h"
45#include "FONgTransform.h"
46
47#include <BESUtil.h>
48#include <BESInternalError.h>
49#include <BESDapError.h>
50#include <BESContextManager.h>
51#include <BESDataDDSResponse.h>
52#include <BESDapNames.h>
53#include <BESDataNames.h>
54#include <BESDebug.h>
55#include <TempFile.h>
56#include <DapFunctionUtils.h>
57
58#include <TheBESKeys.h>
59
60#define JPEG2000_TEMP_DIR "/tmp"
61#define JPEG2000_GCS "WGS84"
62
63string JPEG2000Transmitter::temp_dir;
64string JPEG2000Transmitter::default_gcs;
65
83{
84 // DATA_SERVICE == "dods"
85 add_method(DATA_SERVICE, JPEG2000Transmitter::send_data_as_jp2);
86
87 if (JPEG2000Transmitter::temp_dir.empty()) {
88 // Where is the temp directory for creating these files
89 bool found = false;
90 string key = "JPEG2000.Tempdir";
91 TheBESKeys::TheKeys()->get_value(key, JPEG2000Transmitter::temp_dir, found);
92 if (!found || JPEG2000Transmitter::temp_dir.empty()) {
93 JPEG2000Transmitter::temp_dir = JPEG2000_TEMP_DIR;
94 }
95 string::size_type len = JPEG2000Transmitter::temp_dir.size();
96 if (JPEG2000Transmitter::temp_dir[len - 1] == '/') {
97 JPEG2000Transmitter::temp_dir = JPEG2000Transmitter::temp_dir.substr(0, len - 1);
98 }
99 }
100
101 if (JPEG2000Transmitter::default_gcs.empty()) {
102 // Use what as the default Geographic coordinate system?
103 bool found = false;
104 string key = "JPEG2000.Default_GCS";
105 TheBESKeys::TheKeys()->get_value(key, JPEG2000Transmitter::default_gcs, found);
106 if (!found || JPEG2000Transmitter::default_gcs.empty()) {
107 JPEG2000Transmitter::default_gcs = JPEG2000_GCS;
108 }
109 }
110}
111
128{
129 BESDataDDSResponse *bdds = dynamic_cast<BESDataDDSResponse *>(obj);
130 if (!bdds) {
131 throw BESInternalError("cast error", __FILE__, __LINE__);
132 }
133
134 DDS *dds = bdds->get_dds();
135 if (!dds)
136 throw BESInternalError("No DataDDS has been created for transmit", __FILE__, __LINE__);
137
138 ostream &strm = dhi.get_output_stream();
139 if (!strm)
140 throw BESInternalError("Output stream is not set, cannot return as", __FILE__, __LINE__);
141
142 BESDEBUG("JPEG20002", "JPEG2000Transmitter::send_data - parsing the constraint" << endl);
143
144 // ticket 1248 jhrg 2/23/09
145 string ce = www2id(dhi.data[POST_CONSTRAINT], "%", "%20%26");
146 try {
147 bdds->get_ce().parse_constraint(ce, *dds);
148 }
149 catch (Error &e) {
150 throw BESDapError("Failed to parse the constraint expression: " + e.get_error_message(), false, e.get_error_code(), __FILE__, __LINE__);
151 }
152 catch (...) {
153 throw BESInternalError("Failed to parse the constraint expression: Unknown exception caught", __FILE__, __LINE__);
154 }
155
156 // now we need to read the data
157 BESDEBUG("JPEG20002", "JPEG2000Transmitter::send_data - reading data into DataDDS" << endl);
158
159 bes::TempFile temp_file;
160 string temp_file_name = temp_file.create(JPEG2000Transmitter::temp_dir, "jp2000_");
161#if 0
162 // Huh? Put the template for the temp file name in a char array. Use vector<char>
163 // to avoid using new/delete.
164 string temp_file_name = JPEG2000Transmitter::temp_dir + '/' + "jp2XXXXXX";
165 vector<char> temp_file(temp_file_name.size() + 1);
166 string::size_type len = temp_file_name.copy(temp_file.data(), temp_file_name.size());
167 temp_file[len] = '\0';
168
169 // cover the case where older versions of mkstemp() create the file using
170 // a mode of 666.
171 mode_t original_mode = umask(077);
172
173 // Make and open (an atomic operation) the temporary file. Then reset the umask
174 int fd = mkstemp(temp_file.data());
175 umask(original_mode);
176
177 if (fd == -1)
178 throw BESInternalError("Failed to open the temporary file: " + temp_file_name, __FILE__, __LINE__);
179#endif
180 // transform the OPeNDAP DataDDS to the geotiff file
181 BESDEBUG("JPEG20002", "JPEG2000Transmitter::send_data - transforming into temporary file " << temp_file_name << endl);
182
183 try {
184
185 // Handle *functional* constraint expressions specially
186 if (bdds->get_ce().function_clauses()) {
187 BESDEBUG("fong2", "processing a functional constraint clause(s)." << endl);
188 DDS *tmp_dds = bdds->get_ce().eval_function_clauses(*dds);
189 delete dds;
190 dds = tmp_dds;
191 bdds->set_dds(dds);
192
193 // This next step utilizes a well known function, promote_function_output_structures()
194 // to look for one or more top level Structures whose name indicates (by way of ending
195 // with "_uwrap") that their contents should be promoted (aka moved) to the top level.
196 // This is in support of a hack around the current API where server side functions
197 // may only return a single DAP object and not a collection of objects. The name suffix
198 // "_unwrap" is used as a signal from the function to the the various response
199 // builders and transmitters that the representation needs to be altered before
200 // transmission, and that in fact is what happens in our friend
201 // promote_function_output_structures()
202 promote_function_output_structures(dds);
203
204 }
205 else {
206 // Iterate through the variables in the DataDDS and read
207 // in the data if the variable has the send flag set.
208 for (DDS::Vars_iter i = dds->var_begin(); i != dds->var_end(); i++) {
209 if ((*i)->send_p()) {
210 (*i)->intern_data(bdds->get_ce(), *dds);
211 }
212 }
213 }
214
215 }
216 catch (Error &e) {
217 throw BESDapError("Failed to read data: " + e.get_error_message(), false, e.get_error_code(), __FILE__, __LINE__);
218 }
219 catch (BESError &e) {
220 throw;
221 }
222 catch (...) {
223 throw BESInternalError("Failed to read data: Unknown exception caught", __FILE__, __LINE__);
224 }
225
226 try {
227 FONgTransform ft(dds, bdds->get_ce(), temp_file_name);
228
229 // Now that we are ready to start building the response data we
230 // cancel any pending timeout alarm according to the configuration.
232
233 // transform() opens the temporary file, dumps data to it and closes it.
235
236 BESDEBUG("JPEG20002", "JPEG2000Transmitter::send_data - transmitting temp file " << temp_file_name << endl );
237
238 JPEG2000Transmitter::return_temp_stream(temp_file_name, strm);
239 }
240 catch (Error &e) {
241#if 0
242 close(fd);
243 (void) unlink(temp_file.data());
244#endif
245 throw BESDapError("Failed to transform data to JPEG2000: " + e.get_error_message(), false, e.get_error_code(), __FILE__, __LINE__);
246 }
247 catch (BESError &e) {
248#if 0
249 close(fd);
250 (void) unlink(temp_file.data());
251#endif
252 throw;
253 }
254 catch (...) {
255#if 0
256 close(fd);
257 (void) unlink(temp_file.data());
258#endif
259 throw BESInternalError("Fileout GeoTiff, was not able to transform to JPEG2000, unknown error", __FILE__, __LINE__);
260 }
261
262#if 0
263 close(fd);
264 (void) unlink(temp_file.data());
265#endif
266
267 BESDEBUG("JPEG20002", "JPEG2000Transmitter::send_data - done transmitting to jp2" << endl);
268}
269
279void JPEG2000Transmitter::return_temp_stream(const string &filename, ostream &strm)
280{
281 ifstream os;
282 os.open(filename.c_str(), ios::binary | ios::in);
283 if (!os)
284 throw BESInternalError("Cannot connect to data source", __FILE__, __LINE__);
285
286 char block[4096];
287 os.read(block, sizeof block);
288 int nbytes = os.gcount();
289 if (nbytes == 0) {
290 os.close();
291 throw BESInternalError("Internal server error, got zero count on stream buffer.", __FILE__, __LINE__);
292 }
293
294 bool found = false;
295 string context = "transmit_protocol";
296 string protocol = BESContextManager::TheManager()->get_context(context, found);
297 if (protocol == "HTTP") {
298 strm << "HTTP/1.0 200 OK\n";
299 strm << "Content-type: application/octet-stream\n";
300 strm << "Content-Description: " << "BES dataset" << "\n";
301 strm << "Content-Disposition: filename=" << filename << ".jp2;\n\n";
302 strm << flush;
303 }
304 strm.write(block, nbytes);
305
306 while (os) {
307 os.read(block, sizeof block);
308 nbytes = os.gcount();
309 strm.write(block, nbytes);
310 }
311
312 os.close();
313}
314
virtual std::string get_context(const std::string &name, bool &found)
retrieve the value of the specified context from the BES
error object created from libdap error objects and can handle those errors
Definition BESDapError.h:50
Represents an OPeNDAP DataDDS DAP2 data object within the BES.
void set_dds(libdap::DDS *ddsIn)
Structure storing information used by the BES to handle the request.
std::map< std::string, std::string > data
the map of string data that will be required for the current request.
Base exception class for the BES with basic string message.
Definition BESError.h:66
exception thrown if internal error encountered
Abstract base class representing a specific set of information in response to a request to the BES.
static void conditional_timeout_cancel()
Checks if the timeout alarm should be canceled based on the value of the BES key BES....
Definition BESUtil.cc:898
Transformation object that converts an OPeNDAP DataDDS to a GeoTiff file.
virtual void transform_to_jpeg2000()
Transforms the variables of the DataDDS to a JPEG2000 file.
JPEG2000Transmitter()
Construct the JPEG2000Transmitter, adding it with name geotiff to be able to transmit a data response...
static void send_data_as_jp2(BESResponseObject *obj, BESDataHandlerInterface &dhi)
The static method registered to transmit OPeNDAP data objects as a netcdf file.
void get_value(const std::string &s, std::string &val, bool &found)
Retrieve the value of a given key, if set.
static TheBESKeys * TheKeys()
Access to the singleton.
Definition TheBESKeys.cc:85
Get a new temporary file.
Definition TempFile.h:43
std::string create(const std::string &dir_name="/tmp/hyrax_tmp", const std::string &path_template="opendap")
Create a new temporary file.
Definition TempFile.cc:164
STL class.
STL class.
STL class.