1 /*
2 * $Id$
3 *
4 * Licensed to the Apache Software Foundation (ASF) under one
5 * or more contributor license agreements. See the NOTICE file
6 * distributed with this work for additional information
7 * regarding copyright ownership. The ASF licenses this file
8 * to you under the Apache License, Version 2.0 (the
9 * "License"); you may not use this file except in compliance
10 * with the License. You may obtain a copy of the License at
11 *
12 * http://www.apache.org/licenses/LICENSE-2.0
13 *
14 * Unless required by applicable law or agreed to in writing,
15 * software distributed under the License is distributed on an
16 * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
17 * KIND, either express or implied. See the License for the
18 * specific language governing permissions and limitations
19 * under the License.
20 */
21
22
23 package org.apache.struts.webapp.example.plugin;
24
25
26 import java.io.BufferedInputStream;
27 import java.io.BufferedOutputStream;
28 import java.io.File;
29 import java.io.FileOutputStream;
30 import java.util.ArrayList;
31
32 import jakarta.servlet.ServletException;
33
34 import org.apache.struts.action.ActionServlet;
35 import org.apache.struts.action.PlugIn;
36 import org.apache.struts.apps.mailreader.dao.impl.memory.MemoryUserDatabase;
37 import org.apache.struts.config.ModuleConfig;
38 import org.apache.struts.util.LabelValueBean;
39 import org.apache.struts.webapp.example.Constants;
40 import org.slf4j.Logger;
41 import org.slf4j.LoggerFactory;
42
43 /**
44 * <p><strong>MemoryDatabasePlugIn</strong> initializes and finalizes the
45 * persistent storage of User and Subscription information for the Struts
46 * Demonstration Application, using an in-memory database backed by an
47 * XML file.</p>
48 *
49 * <p><strong>IMPLEMENTATION WARNING</strong> - If this web application is run
50 * from a WAR file, or in another environment where reading and writing of the
51 * web application resource is impossible, the initial contents will be copied
52 * to a file in the web application temporary directory provided by the
53 * container. This is for demonstration purposes only - you should
54 * <strong>NOT</strong> assume that files written here will survive a restart
55 * of your servlet container.</p>
56 *
57 * @author Craig R. McClanahan
58 * @version $Rev$ $Date$
59 */
60
61 public final class MemoryDatabasePlugIn implements PlugIn {
62
63
64 // ----------------------------------------------------- Instance Variables
65
66
67 /**
68 * The {@link MemoryUserDatabase} object we construct and make available.
69 */
70 private MemoryUserDatabase database = null;
71
72
73 /**
74 * The {@code Log} instance for this class.
75 */
76 private final static Logger LOG =
77 LoggerFactory.getLogger(MemoryDatabasePlugIn.class);
78
79
80 /**
81 * The {@link ActionServlet} owning this application.
82 */
83 private ActionServlet servlet = null;
84
85
86 // ------------------------------------------------------------- Properties
87
88
89 /**
90 * The web application resource path of our persistent database
91 * storage file.
92 */
93 private String pathname = "/WEB-INF/database.xml";
94
95 public String getPathname() {
96 return (this.pathname);
97 }
98
99 public void setPathname(String pathname) {
100 this.pathname = pathname;
101 }
102
103
104 // --------------------------------------------------------- PlugIn Methods
105
106
107 /**
108 * Gracefully shut down this database, releasing any resources
109 * that were allocated at initialization.
110 */
111 public void destroy() {
112
113 LOG.info("Finalizing memory database plug in");
114
115 if (database != null) {
116 try {
117 database.close();
118 } catch (Exception e) {
119 LOG.error("Closing memory database", e);
120 }
121 }
122
123 servlet.getServletContext().removeAttribute(Constants.DATABASE_KEY);
124 database = null;
125 servlet = null;
126 database = null;
127 }
128
129
130 /**
131 * Initialize and load our initial database from persistent storage.
132 *
133 * @param servlet The ActionServlet for this web application
134 * @param config The ApplicationConfig for our owning module
135 *
136 * @throws ServletException if we cannot configure ourselves correctly
137 */
138 public void init(ActionServlet servlet, ModuleConfig config)
139 throws ServletException {
140
141 LOG.info("Initializing memory database plug in from '{}'",
142 pathname);
143
144 // Remember our associated configuration and servlet
145 this.servlet = servlet;
146
147 // Construct a new database and make it available
148 database = new MemoryUserDatabase();
149 try {
150 String path = calculatePath();
151 LOG.debug(" Loading database from '{}'", path);
152 database.setPathname(path);
153 database.open();
154 } catch (Exception e) {
155 LOG.error("Opening memory database", e);
156 throw new ServletException("Cannot load database from '" +
157 pathname + "'", e);
158 }
159
160 // Make the initialized database available
161 servlet.getServletContext().setAttribute(Constants.DATABASE_KEY,
162 database);
163
164 // Setup and cache other required data
165 setupCache(servlet, config);
166
167 }
168
169
170 // --------------------------------------------------------- Public Methods
171
172
173 // ------------------------------------------------------ Protected Methods
174
175
176 /**
177 * <p>Cache commonly required data as servlet context attributes.</p>
178 *
179 * @param servlet The <code>ActionServlet</code> instance running
180 * this webapp
181 * @param config The <code>ModuleConfig</code> for this application module
182 */
183 protected void setupCache(ActionServlet servlet, ModuleConfig config) {
184
185 // Set up list of server types under "serverTypes"
186 ArrayList<LabelValueBean> serverTypes = new ArrayList<>();
187 serverTypes.add(new LabelValueBean("IMAP Protocol", "imap"));
188 serverTypes.add(new LabelValueBean("POP3 Protocol", "pop3"));
189 servlet.getServletContext().setAttribute("serverTypes", serverTypes);
190
191 }
192
193
194
195
196 // -------------------------------------------------------- Private Methods
197
198
199 /**
200 * Calculate and return an absolute pathname to the XML file to contain
201 * our persistent storage information.
202 *
203 * @throws Exception if an input/output error occurs
204 */
205 private String calculatePath() throws Exception {
206
207 // Can we access the database via file I/O?
208 String path = servlet.getServletContext().getRealPath(pathname);
209 if (path != null) {
210 return (path);
211 }
212
213 // Does a copy of this file already exist in our temporary directory
214 File dir = (File)
215 servlet.getServletContext().getAttribute
216 ("jakarta.servlet.context.tempdir");
217 File file = new File(dir, "struts-example-database.xml");
218 if (file.exists()) {
219 return (file.getAbsolutePath());
220 }
221
222 // Copy the static resource to a temporary file and return its path
223 try (BufferedInputStream bis = new BufferedInputStream(
224 servlet.getServletContext().getResourceAsStream(pathname), 1024);
225 BufferedOutputStream bos = new BufferedOutputStream(new FileOutputStream(file), 1024)) {
226
227 byte buffer[] = new byte[1024];
228 while (true) {
229 int n = bis.read(buffer);
230 if (n <= 0) {
231 break;
232 }
233 bos.write(buffer, 0, n);
234 }
235 }
236
237 return (file.getAbsolutePath());
238
239 }
240 }