All Downloads are FREE. Search and download functionalities are using the official Maven repository.

marytts.server.http.MaryHttpServer Maven / Gradle / Ivy

The newest version!
/**
 * Copyright 2000-2006 DFKI GmbH.
 * All Rights Reserved.  Use is subject to license terms.
 *
 * This file is part of MARY TTS.
 *
 * MARY TTS is free software: you can redistribute it and/or modify
 * it under the terms of the GNU Lesser General Public License as published by
 * the Free Software Foundation, version 3 of the License.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public License
 * along with this program.  If not, see .
 *
 */
package marytts.server.http;

import java.io.IOException;
import java.io.InterruptedIOException;
import java.net.InetSocketAddress;

import marytts.server.MaryProperties;
import marytts.util.MaryUtils;

import org.apache.http.HttpException;
import org.apache.http.impl.DefaultConnectionReuseStrategy;
import org.apache.http.impl.DefaultHttpResponseFactory;
import org.apache.http.impl.nio.DefaultServerIOEventDispatch;
import org.apache.http.impl.nio.reactor.DefaultListeningIOReactor;
import org.apache.http.nio.NHttpConnection;
import org.apache.http.nio.protocol.BufferingHttpServiceHandler;
import org.apache.http.nio.protocol.EventListener;
import org.apache.http.nio.reactor.IOEventDispatch;
import org.apache.http.nio.reactor.ListeningIOReactor;
import org.apache.http.params.BasicHttpParams;
import org.apache.http.params.CoreConnectionPNames;
import org.apache.http.params.CoreProtocolPNames;
import org.apache.http.params.HttpParams;
import org.apache.http.protocol.BasicHttpProcessor;
import org.apache.http.protocol.ExecutionContext;
import org.apache.http.protocol.HttpRequestHandlerRegistry;
import org.apache.http.protocol.ResponseConnControl;
import org.apache.http.protocol.ResponseContent;
import org.apache.http.protocol.ResponseDate;
import org.apache.http.protocol.ResponseServer;
import org.apache.log4j.Logger;

/**
 * Listen for clients as an Http server at port MaryProperties.socketPort().
 * 

* There are two types of clients that can be handled: *

* (1) Non-web browser clients (2) Web browser clients *

* Note that non-web browser clients can mimic web browser clients by setting WEB_BROWSER_CLIENT parameter to "true" in the Http * request string *

* Clients can request the following (See below for more details): *

* (1) A file such as Mary icon or an audio file (2) Information like available voices, example texts, available audio formats, * etc (3) Synthesis of an appropriate input with appropriate additional parameters *

* For all clients, the responses are always sent in an HttpResponse. The entity in the response body can represent: *

* (1) An html page (applies only to web browser clients) (2) Some binary data (such as bytes of Mary icon for file requests, or * bytes of audio data for synthesis requests) (3) Some piece of text *

* A valid Mary Http request string is a collection of individual key-value pairs combined in Http request style: * address?pair1&pair2&pair3... etc. *

* The address identifies the kind of thing that the client is asking for: *

    *
  • version requests the version of the MARY server;
  • *
  • datatypes requests the list of available data types;
  • *
  • locales requests the list of available locales / language components;
  • *
  • voices requests the list of available voices;
  • *
  • audioformats requests the list of supported audio file format types;
  • *
  • exampletext?voice=hmm-slt requests the example text for the given voice;
  • *
  • exampletext?datatype=RAWMARYXML&locale=de requests an example text for data of the given type and locale;
  • *
  • audioeffects requests the list of default audio effects;
  • *
  • audioeffect-default-param?effect=Robot requests the default parameters of the given audio effect;
  • *
  • audioeffect-full?effect=Robot&params=amount:100.0 requests a full description of the given audio effect, * including effect name, parameters and help text;
  • *
  • audioeffect-help?effect=Robot requests a help text describing the given audio effect;
  • *
  • audioeffect-is-hmm-effect?effect=Robot requests a boolean value (plain text "yes" or "no") indicating whether * or not the given effect is an effect that operates on HMM-based voices only;
  • *
  • features?locale=de requests the list of available features that can be computed for the given locale;
  • *
  • features?voice=hmm-slt requests the list of available features that can be computed for the given voice;
  • *
  • vocalizations?voice=dfki-poppy requests the list of vocalization names that are available with the given * voice; *
  • styles?voice=dfki-pavoque-styles requests the list of style names that are available with the given voice; *
  • process requests the synthesis of some text (see below).
  • *
*

* In Each pair has the following structure: *

* KEY=VALUE *

* where the following keys are used for passing additional information from server to client and/or vice versa: *

* INPUT_TYPE (input data type) *

* OUTPUT_TYPE (output data type) *

* AUDIO (audio format. It may include streaming/non-streaming information as well. Example values for non-streaming formats: * AU_FILE, MP3_FILE, WAVE_FILE Example values for streaming formats: AU_STREAM, MP3_STREAM) *

* STYLE (Style descriptor) *

* INPUT_TEXT (Input text to be synthesised) *

* OUTPUT_TEXT (Output text - if the output type is not audio) *

* SYNTHESIS_OUTPUT (A key to ask for synthesis, or to represent synthesis result. Example values: SYNTHESIS_OUTPUT=? instantiates * a synthesis request In response, the server can set SYNTHESIS_OUTPUT to DONE, PENDING, or FAILED depending on the validity and * type of te request PENDING is a special case used for handling double requests due to EMBED or OBJECT tags in web browser * client html pages * *

* Additionally, web browser clients should use the following key-value pair to tell the server about their type: *

* WEB_BROWSER_CLIENT=true (All other values will be interpreted as non-web browser client) *

* An easy way to test the http server is as follows: *

* (1) Run mary server in "http" mode by setting server=http in marybase.config *

* (2) Copy and paste the following to a web browser´s address bar: *

* marytts server intro *

* Provided that the server runs at localhost:59125 (or change "http://localhost:59125/" part as required), the web browser * supports AUDIO type (if not try other formats such as WAVE, MP3, OGG or install a plug-in to play the target format), and the * VOICE is installed (hmm-slt), the synthesis result should be sent to the web browser for playback or saving (depending on web * browser settings). *

* * check {@link InfoRequestHandler}, {@link FileRequestHandler}, {@link SynthesisRequestHandler} . * * @author Oytun Türk, Marc Schröder */ public class MaryHttpServer extends Thread { private static Logger logger; private boolean isReady = false; public MaryHttpServer() { logger = MaryUtils.getLogger("server"); } public boolean isReady() { return isReady; } public void run() { logger.info("Starting server."); int localPort = MaryProperties.needInteger("socket.port"); HttpParams params = new BasicHttpParams(); params.setIntParameter(CoreConnectionPNames.SO_TIMEOUT, 0) // 0 means no timeout, any positive value means time out in miliseconds (i.e. 50000 for 50 seconds) .setIntParameter(CoreConnectionPNames.SOCKET_BUFFER_SIZE, 8 * 1024) .setBooleanParameter(CoreConnectionPNames.STALE_CONNECTION_CHECK, false) .setBooleanParameter(CoreConnectionPNames.TCP_NODELAY, true) .setParameter(CoreProtocolPNames.ORIGIN_SERVER, "HttpComponents/1.1"); BasicHttpProcessor httpproc = new BasicHttpProcessor(); httpproc.addInterceptor(new ResponseDate()); httpproc.addInterceptor(new ResponseServer()); httpproc.addInterceptor(new ResponseContent()); httpproc.addInterceptor(new ResponseConnControl()); BufferingHttpServiceHandler handler = new BufferingHttpServiceHandler(httpproc, new DefaultHttpResponseFactory(), new DefaultConnectionReuseStrategy(), params); // Set up request handlers HttpRequestHandlerRegistry registry = new HttpRequestHandlerRegistry(); registry.register("/process", new SynthesisRequestHandler()); InfoRequestHandler infoRH = new InfoRequestHandler(); registry.register("/version", infoRH); registry.register("/datatypes", infoRH); registry.register("/locales", infoRH); registry.register("/voices", infoRH); registry.register("/audioformats", infoRH); registry.register("/exampletext", infoRH); registry.register("/audioeffects", infoRH); registry.register("/audioeffect-default-param", infoRH); registry.register("/audioeffect-full", infoRH); registry.register("/audioeffect-help", infoRH); registry.register("/audioeffect-is-hmm-effect", infoRH); registry.register("/features", infoRH); registry.register("/features-discrete", infoRH); registry.register("/vocalizations", infoRH); registry.register("/styles", infoRH); registry.register("*", new FileRequestHandler()); handler.setHandlerResolver(registry); // Provide an event logger handler.setEventListener(new EventLogger()); IOEventDispatch ioEventDispatch = new DefaultServerIOEventDispatch(handler, params); int numParallelThreads = MaryProperties.getInteger("server.http.parallelthreads", 5); logger.info("Waiting for client to connect on port " + localPort); try { ListeningIOReactor ioReactor = new DefaultListeningIOReactor(numParallelThreads, params); ioReactor.listen(new InetSocketAddress(localPort)); isReady = true; ioReactor.execute(ioEventDispatch); } catch (InterruptedIOException ex) { logger.info("Interrupted", ex); } catch (IOException e) { logger.info("Problem with HTTP connection", e); } logger.debug("Shutdown"); } static class EventLogger implements EventListener { public void connectionOpen(final NHttpConnection conn) { logger.info("Connection from " + conn.getContext().getAttribute(ExecutionContext.HTTP_TARGET_HOST) // conn.getInetAddress().getHostName() ); } public void connectionTimeout(final NHttpConnection conn) { logger.info("Connection timed out: " + conn); } public void connectionClosed(final NHttpConnection conn) { logger.info("Connection closed: " + conn); } public void fatalIOException(final IOException ex, final NHttpConnection conn) { logger.info("I/O error: " + ex.getMessage()); } public void fatalProtocolException(final HttpException ex, final NHttpConnection conn) { logger.info("HTTP error: " + ex.getMessage()); } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy