Compare commits

...

16 Commits

Author SHA1 Message Date
494bc6b6a4 csharp: remove storage service 2017-10-09 17:39:51 +03:00
72db6178ac java: remove storage service 2017-10-09 17:35:12 +03:00
5c1d925d42 nodejs: remove storage service 2017-10-09 17:33:30 +03:00
fd4dfea7d7 php: remove storage service 2017-10-09 17:31:49 +03:00
ebd309dabc ruby: remove storage service 2017-10-09 17:29:09 +03:00
f376c50fd9 try convert once 2017-10-09 17:16:24 +03:00
6088e7d685 ruby: gemfile up 2017-10-09 17:01:43 +03:00
d143e3c80d ruby: json response 2017-10-09 16:24:09 +03:00
56d6388603 ruby: change track event 2017-10-09 15:47:21 +03:00
54993942bf java: json response 2017-10-06 16:06:41 +03:00
f2301ecbbf java: fix opening editor 2017-10-06 16:06:01 +03:00
f09f62103c java: formatting 2017-10-06 16:05:44 +03:00
e1e3357748 nodejs: json response 2017-09-28 18:40:18 +03:00
449020a9a4 php: json response 2017-09-28 18:18:23 +03:00
22f2f0f922 mvc: json response 2017-09-28 17:55:01 +03:00
1aa7e73d79 csharp: json response 2017-09-28 17:47:54 +03:00
24 changed files with 438 additions and 1011 deletions

View File

@ -29,10 +29,8 @@ using System.IO;
using System.Net;
using System.Text;
using System.Text.RegularExpressions;
using System.Web;
using System.Web.Configuration;
using System.Xml;
using System.Xml.Linq;
using System.Web.Helpers;
namespace OnlineEditorsExampleMVC.Helpers
{
@ -47,7 +45,6 @@ namespace OnlineEditorsExampleMVC.Helpers
static ServiceConverter()
{
DocumentConverterUrl = WebConfigurationManager.AppSettings["files.docservice.url.converter"] ?? "";
DocumentStorageUrl = WebConfigurationManager.AppSettings["files.docservice.url.storage"] ?? "";
Int32.TryParse(WebConfigurationManager.AppSettings["files.docservice.timeout"], out ConvertTimeout);
ConvertTimeout = ConvertTimeout > 0 ? ConvertTimeout : 120000;
@ -65,21 +62,6 @@ namespace OnlineEditorsExampleMVC.Helpers
/// </summary>
private static readonly string DocumentConverterUrl;
/// <summary>
/// Url to the service of storage
/// </summary>
private static readonly string DocumentStorageUrl;
/// <summary>
/// The parameters for the query conversion
/// </summary>
private const string ConvertParams = "?url={0}&outputtype={1}&filetype={2}&title={3}&key={4}";
/// <summary>
/// Number of tries request conversion
/// </summary>
private const int MaxTry = 3;
#endregion
#region public method
@ -108,74 +90,51 @@ namespace OnlineEditorsExampleMVC.Helpers
out string convertedDocumentUri)
{
convertedDocumentUri = string.Empty;
var responceFromConvertService =
SendRequestToConvertService(documentUri, fromExtension, toExtension, documentRevisionId, isAsync)
.Root;
var errorElement = responceFromConvertService.Element("Error");
if (errorElement != null)
ProcessConvertServiceResponceError(Convert.ToInt32(errorElement.Value));
fromExtension = string.IsNullOrEmpty(fromExtension) ? Path.GetExtension(documentUri) : fromExtension;
var isEndConvert = Convert.ToBoolean(responceFromConvertService.Element("EndConvert").Value);
var percent = Convert.ToInt32(responceFromConvertService.Element("Percent").Value);
var title = Path.GetFileName(documentUri);
title = string.IsNullOrEmpty(title) ? Guid.NewGuid().ToString() : title;
if (isEndConvert)
{
convertedDocumentUri = responceFromConvertService.Element("FileUrl").Value;
percent = 100;
}
else
{
percent = percent >= 100 ? 99 : percent;
}
documentRevisionId = string.IsNullOrEmpty(documentRevisionId)
? documentUri
: documentRevisionId;
documentRevisionId = GenerateRevisionId(documentRevisionId);
return percent;
}
/// <summary>
/// Placing the document in the storage service
/// </summary>
/// <param name="fileStream">Stream of document</param>
/// <param name="contentLength">Length of stream</param>
/// <param name="contentType">Mime type</param>
/// <param name="documentRevisionId">Key for caching on service, whose used in editor</param>
/// <returns>Uri to document in the storage</returns>
public static string GetExternalUri(
Stream fileStream,
long contentLength,
string contentType,
string documentRevisionId)
{
var urlDocumentService = DocumentStorageUrl + ConvertParams;
var urlTostorage = String.Format(urlDocumentService,
string.Empty,
string.Empty,
string.Empty,
string.Empty,
documentRevisionId);
var request = (HttpWebRequest)WebRequest.Create(urlTostorage);
var request = (HttpWebRequest)WebRequest.Create(DocumentConverterUrl);
request.Method = "POST";
request.ContentType = contentType;
request.ContentLength = contentLength;
request.ContentType = "application/json";
request.Accept = "application/json";
request.Timeout = ConvertTimeout;
const int bufferSize = 2048;
var buffer = new byte[bufferSize];
int readed;
while ((readed = fileStream.Read(buffer, 0, bufferSize)) > 0)
var bodyString = string.Format("{{\"async\": {0},\"filetype\": \"{1}\",\"key\": \"{2}\",\"outputtype\": \"{3}\",\"title\": \"{4}\",\"url\": \"{5}\"}}",
isAsync.ToString().ToLower(),
fromExtension.Trim('.'),
documentRevisionId,
toExtension.Trim('.'),
title,
documentUri);
var bytes = Encoding.UTF8.GetBytes(bodyString);
request.ContentLength = bytes.Length;
using (var requestStream = request.GetRequestStream())
{
request.GetRequestStream().Write(buffer, 0, readed);
requestStream.Write(bytes, 0, bytes.Length);
}
string dataResponse;
using (var response = request.GetResponse())
using (var stream = response.GetResponseStream())
{
if (stream == null) throw new WebException("Could not get an answer");
var xDocumentResponse = XDocument.Load(new XmlTextReader(stream));
string externalUri;
GetResponseUri(xDocumentResponse, out externalUri);
return externalUri;
if (stream == null) throw new Exception("Response is null");
using (var reader = new StreamReader(stream))
{
dataResponse = reader.ReadToEnd();
}
}
return GetResponseUri(dataResponse, out convertedDocumentUri);
}
/// <summary>
@ -195,77 +154,44 @@ namespace OnlineEditorsExampleMVC.Helpers
#region private method
/// <summary>
/// Request for conversion to a service
/// Processing document received from the editing service
/// </summary>
/// <param name="documentUri">Uri for the document to convert</param>
/// <param name="fromExtension">Document extension</param>
/// <param name="toExtension">Extension to which to convert</param>
/// <param name="documentRevisionId">Key for caching on service</param>
/// <param name="isAsync">Perform conversions asynchronously</param>
/// <returns>Xml document request result of conversion</returns>
private static XDocument SendRequestToConvertService(string documentUri, string fromExtension, string toExtension, string documentRevisionId, bool isAsync)
/// <param name="jsonDocumentResponse">The resulting json from editing service</param>
/// <param name="responseUri">Uri to the converted document</param>
/// <returns>The percentage of completion of conversion</returns>
private static int GetResponseUri(string jsonDocumentResponse, out string responseUri)
{
fromExtension = string.IsNullOrEmpty(fromExtension) ? Path.GetExtension(documentUri) : fromExtension;
if (string.IsNullOrEmpty(jsonDocumentResponse)) throw new ArgumentException("Invalid param", "jsonDocumentResponse");
var title = Path.GetFileName(documentUri);
title = string.IsNullOrEmpty(title) ? Guid.NewGuid().ToString() : title;
var responseFromService = Json.Decode(jsonDocumentResponse);
if (jsonDocumentResponse == null) throw new WebException("Invalid answer format");
documentRevisionId = string.IsNullOrEmpty(documentRevisionId)
? documentUri
: documentRevisionId;
documentRevisionId = GenerateRevisionId(documentRevisionId);
var errorElement = responseFromService.error;
if (errorElement != null) ProcessResponseError(Convert.ToInt32(errorElement));
var req = (HttpWebRequest)WebRequest.Create(DocumentConverterUrl);
req.Method = "POST";
req.ContentType = "text/json";
req.Timeout = ConvertTimeout;
var isEndConvert = responseFromService.endConvert;
var bodyString = string.Format("{{\"async\": {0},\"filetype\": \"{1}\",\"key\": \"{2}\",\"outputtype\": \"{3}\",\"title\": \"{4}\",\"url\": \"{5}\"}}",
isAsync.ToString().ToLower(),
fromExtension.Trim('.'),
documentRevisionId,
toExtension.Trim('.'),
title,
documentUri);
var bytes = Encoding.UTF8.GetBytes(bodyString);
req.ContentLength = bytes.Length;
using (var requestStream = req.GetRequestStream())
int resultPercent;
responseUri = string.Empty;
if (isEndConvert)
{
requestStream.Write(bytes, 0, bytes.Length);
responseUri = responseFromService.fileUrl;
resultPercent = 100;
}
else
{
resultPercent = responseFromService.percent;
if (resultPercent >= 100) resultPercent = 99;
}
Stream stream = null;
var countTry = 0;
while (countTry < MaxTry)
{
try
{
countTry++;
stream = req.GetResponse().GetResponseStream();
break;
}
catch (WebException ex)
{
if (ex.Status != WebExceptionStatus.Timeout)
{
throw new HttpException((int) HttpStatusCode.BadRequest, "Bad Request", ex);
}
}
}
if (countTry == MaxTry)
{
throw new WebException("Timeout", WebExceptionStatus.Timeout);
}
return XDocument.Load(new XmlTextReader(stream));
return resultPercent;
}
/// <summary>
/// Generate an error code table
/// </summary>
/// <param name="errorCode">Error code</param>
private static void ProcessConvertServiceResponceError(int errorCode)
private static void ProcessResponseError(int errorCode)
{
var errorMessage = string.Empty;
const string errorMessageTemplate = "Error occurred in the ConvertService.ashx: {0}";
@ -315,45 +241,6 @@ namespace OnlineEditorsExampleMVC.Helpers
throw new Exception(errorMessage);
}
/// <summary>
/// Processing document received from the editing service
/// </summary>
/// <param name="xDocumentResponse">The resulting xml from editing service</param>
/// <param name="responseUri">Uri to the converted document</param>
/// <returns>The percentage of completion of conversion</returns>
private static int GetResponseUri(XDocument xDocumentResponse, out string responseUri)
{
var responceFromConvertService = xDocumentResponse.Root;
if (responceFromConvertService == null) throw new WebException("Invalid answer format");
var errorElement = responceFromConvertService.Element("Error");
if (errorElement != null) ProcessConvertServiceResponceError(Convert.ToInt32(errorElement.Value));
var endConvert = responceFromConvertService.Element("EndConvert");
if (endConvert == null) throw new WebException("Invalid answer format");
var isEndConvert = Convert.ToBoolean(endConvert.Value);
var resultPercent = 0;
responseUri = string.Empty;
if (isEndConvert)
{
var fileUrl = responceFromConvertService.Element("FileUrl");
if (fileUrl == null) throw new WebException("Invalid answer format");
responseUri = fileUrl.Value;
resultPercent = 100;
}
else
{
var percent = responceFromConvertService.Element("Percent");
if (percent != null)
resultPercent = Convert.ToInt32(percent.Value);
resultPercent = resultPercent >= 100 ? 99 : resultPercent;
}
return resultPercent;
}
#endregion
}
}

View File

@ -49,12 +49,10 @@
<Reference Include="System.ComponentModel.DataAnnotations" />
<Reference Include="System.Core" />
<Reference Include="System.Data.DataSetExtensions" />
<Reference Include="System.Xml.Linq" />
<Reference Include="System.Web" />
<Reference Include="System.Web.Extensions" />
<Reference Include="System.Web.Abstractions" />
<Reference Include="System.Web.Routing" />
<Reference Include="System.Xml" />
<Reference Include="System.Configuration" />
<Reference Include="System.Web.Services" />
<Reference Include="System.EnterpriseServices" />

View File

@ -14,7 +14,6 @@
<add key="files.docservice.convert-docs" value=".docm|.dotx|.dotm|.dot|.doc|.odt|.fodt|.xlsm|.xltx|.xltm|.xlt|.xls|.ods|.fods|.pptm|.ppt|.ppsm|.pps|.potx|.potm|.pot|.odp|.fodp|.rtf|.mht|.html|.htm|.epub"/>
<add key="files.docservice.timeout" value="120000" />
<add key="files.docservice.url.storage" value="https://doc.onlyoffice.com/FileUploader.ashx"/>
<add key="files.docservice.url.converter" value="https://doc.onlyoffice.com/ConvertService.ashx"/>
<add key="files.docservice.url.api" value="https://doc.onlyoffice.com/web-apps/apps/api/documents/api.js"/>
<add key="files.docservice.url.preloader" value="https://doc.onlyoffice.com/web-apps/apps/api/documents/cache-scripts.html"/>

View File

@ -24,16 +24,14 @@
*
*/
using System.Text;
using OnlineEditorsExample;
using System;
using System.IO;
using System.Net;
using System.Text;
using System.Text.RegularExpressions;
using System.Web;
using System.Web.Configuration;
using System.Xml;
using System.Xml.Linq;
using System.Web.Helpers;
using OnlineEditorsExample;
namespace ASC.Api.DocumentConverter
{
@ -48,7 +46,6 @@ namespace ASC.Api.DocumentConverter
static ServiceConverter()
{
DocumentConverterUrl = WebConfigurationManager.AppSettings["files.docservice.url.converter"] ?? "";
DocumentStorageUrl = WebConfigurationManager.AppSettings["files.docservice.url.storage"] ?? "";
Int32.TryParse(WebConfigurationManager.AppSettings["files.docservice.timeout"], out ConvertTimeout);
ConvertTimeout = ConvertTimeout > 0 ? ConvertTimeout : 120000;
@ -66,21 +63,6 @@ namespace ASC.Api.DocumentConverter
/// </summary>
private static readonly string DocumentConverterUrl;
/// <summary>
/// Url to the service of storage
/// </summary>
private static readonly string DocumentStorageUrl;
/// <summary>
/// The parameters for the query conversion
/// </summary>
private const string ConvertParams = "?url={0}&outputtype={1}&filetype={2}&title={3}&key={4}";
/// <summary>
/// Number of tries request conversion
/// </summary>
private const int MaxTry = 3;
#endregion
#region public method
@ -109,63 +91,36 @@ namespace ASC.Api.DocumentConverter
out string convertedDocumentUri)
{
convertedDocumentUri = string.Empty;
var responceFromConvertService =
SendRequestToConvertService(documentUri, fromExtension, toExtension, documentRevisionId, isAsync)
.Root;
var errorElement = responceFromConvertService.Element("Error");
if (errorElement != null)
ProcessConvertServiceResponceError(Convert.ToInt32(errorElement.Value));
fromExtension = string.IsNullOrEmpty(fromExtension) ? Path.GetExtension(documentUri) : fromExtension;
var isEndConvert = Convert.ToBoolean(responceFromConvertService.Element("EndConvert").Value);
var percent = Convert.ToInt32(responceFromConvertService.Element("Percent").Value);
var title = Path.GetFileName(documentUri);
title = string.IsNullOrEmpty(title) ? Guid.NewGuid().ToString() : title;
if (isEndConvert)
{
convertedDocumentUri = responceFromConvertService.Element("FileUrl").Value;
percent = 100;
}
else
{
percent = percent >= 100 ? 99 : percent;
}
documentRevisionId = string.IsNullOrEmpty(documentRevisionId)
? documentUri
: documentRevisionId;
documentRevisionId = GenerateRevisionId(documentRevisionId);
return percent;
}
/// <summary>
/// Placing the document in the storage service
/// </summary>
/// <param name="fileStream">Stream of document</param>
/// <param name="contentLength">Length of stream</param>
/// <param name="contentType">Mime type</param>
/// <param name="documentRevisionId">Key for caching on service, whose used in editor</param>
/// <returns>Uri to document in the storage</returns>
public static string GetExternalUri(
Stream fileStream,
long contentLength,
string contentType,
string documentRevisionId)
{
var urlDocumentService = DocumentStorageUrl + ConvertParams;
var urlTostorage = String.Format(urlDocumentService,
string.Empty,
string.Empty,
string.Empty,
string.Empty,
documentRevisionId);
var request = (HttpWebRequest)WebRequest.Create(urlTostorage);
var request = (HttpWebRequest)WebRequest.Create(DocumentConverterUrl);
request.Method = "POST";
request.ContentType = contentType;
request.ContentLength = contentLength;
request.ContentType = "application/json";
request.Accept = "application/json";
request.Timeout = ConvertTimeout;
const int bufferSize = 2048;
var buffer = new byte[bufferSize];
int readed;
while ((readed = fileStream.Read(buffer, 0, bufferSize)) > 0)
var bodyString = string.Format("{{\"async\": {0},\"filetype\": \"{1}\",\"key\": \"{2}\",\"outputtype\": \"{3}\",\"title\": \"{4}\",\"url\": \"{5}\"}}",
isAsync.ToString().ToLower(),
fromExtension.Trim('.'),
documentRevisionId,
toExtension.Trim('.'),
title,
documentUri);
var bytes = Encoding.UTF8.GetBytes(bodyString);
request.ContentLength = bytes.Length;
using (var requestStream = request.GetRequestStream())
{
request.GetRequestStream().Write(buffer, 0, readed);
requestStream.Write(bytes, 0, bytes.Length);
}
// hack. http://ubuntuforums.org/showthread.php?t=1841740
@ -174,15 +129,19 @@ namespace ASC.Api.DocumentConverter
ServicePointManager.ServerCertificateValidationCallback += (s, ce, ca, p) => true;
}
string dataResponse;
using (var response = request.GetResponse())
using (var stream = response.GetResponseStream())
{
if (stream == null) throw new WebException("Could not get an answer");
var xDocumentResponse = XDocument.Load(new XmlTextReader(stream));
string externalUri;
GetResponseUri(xDocumentResponse, out externalUri);
return externalUri;
if (stream == null) throw new Exception("Response is null");
using (var reader = new StreamReader(stream))
{
dataResponse = reader.ReadToEnd();
}
}
return GetResponseUri(dataResponse, out convertedDocumentUri);
}
/// <summary>
@ -200,85 +159,46 @@ namespace ASC.Api.DocumentConverter
#endregion
#region private method
/// <summary>
/// Request for conversion to a service
/// Processing document received from the editing service
/// </summary>
/// <param name="documentUri">Uri for the document to convert</param>
/// <param name="fromExtension">Document extension</param>
/// <param name="toExtension">Extension to which to convert</param>
/// <param name="documentRevisionId">Key for caching on service</param>
/// <param name="isAsync">Perform conversions asynchronously</param>
/// <returns>Xml document request result of conversion</returns>
private static XDocument SendRequestToConvertService(string documentUri, string fromExtension, string toExtension, string documentRevisionId, bool isAsync)
/// <param name="jsonDocumentResponse">The resulting json from editing service</param>
/// <param name="responseUri">Uri to the converted document</param>
/// <returns>The percentage of completion of conversion</returns>
private static int GetResponseUri(string jsonDocumentResponse, out string responseUri)
{
fromExtension = string.IsNullOrEmpty(fromExtension) ? Path.GetExtension(documentUri) : fromExtension;
if (string.IsNullOrEmpty(jsonDocumentResponse)) throw new ArgumentException("Invalid param", "jsonDocumentResponse");
var title = Path.GetFileName(documentUri);
title = string.IsNullOrEmpty(title) ? Guid.NewGuid().ToString() : title;
var responseFromService = Json.Decode(jsonDocumentResponse);
if (jsonDocumentResponse == null) throw new WebException("Invalid answer format");
documentRevisionId = string.IsNullOrEmpty(documentRevisionId)
? documentUri
: documentRevisionId;
documentRevisionId = GenerateRevisionId(documentRevisionId);
var errorElement = responseFromService.error;
if (errorElement != null) ProcessResponseError(Convert.ToInt32(errorElement));
var req = (HttpWebRequest)WebRequest.Create(DocumentConverterUrl);
req.Method = "POST";
req.ContentType = "text/json";
req.Timeout = ConvertTimeout;
var isEndConvert = responseFromService.endConvert;
var bodyString = string.Format("{{\"async\": {0},\"filetype\": \"{1}\",\"key\": \"{2}\",\"outputtype\": \"{3}\",\"title\": \"{4}\",\"url\": \"{5}\"}}",
isAsync.ToString().ToLower(),
fromExtension.Trim('.'),
documentRevisionId,
toExtension.Trim('.'),
title,
documentUri);
var bytes = Encoding.UTF8.GetBytes(bodyString);
req.ContentLength = bytes.Length;
using (var requestStream = req.GetRequestStream())
int resultPercent;
responseUri = string.Empty;
if (isEndConvert)
{
requestStream.Write(bytes, 0, bytes.Length);
responseUri = responseFromService.fileUrl;
resultPercent = 100;
}
else
{
resultPercent = responseFromService.percent;
if (resultPercent >= 100) resultPercent = 99;
}
var countTry = 0;
Stream stream = null;
// hack. http://ubuntuforums.org/showthread.php?t=1841740
if (_Default.IsMono)
{
ServicePointManager.ServerCertificateValidationCallback += (s, ce, ca, p) => true;
}
while (countTry < MaxTry)
{
try
{
countTry++;
stream = req.GetResponse().GetResponseStream();
break;
}
catch (WebException ex)
{
if (ex.Status != WebExceptionStatus.Timeout)
{
throw new HttpException((int) HttpStatusCode.BadRequest, "Bad Request", ex);
}
}
}
if (countTry == MaxTry)
{
throw new WebException("Timeout", WebExceptionStatus.Timeout);
}
return XDocument.Load(new XmlTextReader(stream));
return resultPercent;
}
/// <summary>
/// Generate an error code table
/// </summary>
/// <param name="errorCode">Error code</param>
private static void ProcessConvertServiceResponceError(int errorCode)
private static void ProcessResponseError(int errorCode)
{
var errorMessage = string.Empty;
const string errorMessageTemplate = "Error occurred in the ConvertService.ashx: {0}";
@ -328,45 +248,6 @@ namespace ASC.Api.DocumentConverter
throw new Exception(errorMessage);
}
/// <summary>
/// Processing document received from the editing service
/// </summary>
/// <param name="xDocumentResponse">The resulting xml from editing service</param>
/// <param name="responseUri">Uri to the converted document</param>
/// <returns>The percentage of completion of conversion</returns>
private static int GetResponseUri(XDocument xDocumentResponse, out string responseUri)
{
var responceFromConvertService = xDocumentResponse.Root;
if (responceFromConvertService == null) throw new WebException("Invalid answer format");
var errorElement = responceFromConvertService.Element("Error");
if (errorElement != null) ProcessConvertServiceResponceError(Convert.ToInt32(errorElement.Value));
var endConvert = responceFromConvertService.Element("EndConvert");
if (endConvert == null) throw new WebException("Invalid answer format");
var isEndConvert = Convert.ToBoolean(endConvert.Value);
var resultPercent = 0;
responseUri = string.Empty;
if (isEndConvert)
{
var fileUrl = responceFromConvertService.Element("FileUrl");
if (fileUrl == null) throw new WebException("Invalid answer format");
responseUri = fileUrl.Value;
resultPercent = 100;
}
else
{
var percent = responceFromConvertService.Element("Percent");
if (percent != null)
resultPercent = Convert.ToInt32(percent.Value);
resultPercent = resultPercent >= 100 ? 99 : resultPercent;
}
return resultPercent;
}
#endregion
}
}

View File

@ -44,12 +44,12 @@
<Prefer32Bit>false</Prefer32Bit>
</PropertyGroup>
<ItemGroup>
<Reference Include="Microsoft.CSharp" />
<Reference Include="System" />
<Reference Include="System.Web.Extensions" />
<Reference Include="System.Web" />
<Reference Include="System.Xml" />
<Reference Include="System.Web.Helpers" />
<Reference Include="System.Web.Services" />
<Reference Include="System.Xml.Linq" />
</ItemGroup>
<ItemGroup>
<Content Include="App_Themes\images\file_docx.png" />

View File

@ -10,7 +10,6 @@
<add key="files.docservice.convert-docs" value=".docm|.dotx|.dotm|.dot|.doc|.odt|.fodt|.xlsm|.xltx|.xltm|.xlt|.xls|.ods|.fods|.pptm|.ppt|.ppsm|.pps|.potx|.potm|.pot|.odp|.fodp|.rtf|.mht|.html|.htm|.epub"/>
<add key="files.docservice.timeout" value="120000" />
<add key="files.docservice.url.storage" value="https://doc.onlyoffice.com/FileUploader.ashx"/>
<add key="files.docservice.url.converter" value="https://doc.onlyoffice.com/ConvertService.ashx"/>
<add key="files.docservice.url.api" value="https://doc.onlyoffice.com/web-apps/apps/api/documents/api.js"/>
<add key="files.docservice.url.preloader" value="https://doc.onlyoffice.com/web-apps/apps/api/documents/cache-scripts.html"/>

View File

@ -39,8 +39,8 @@ import entities.FileModel;
@WebServlet(name = "EditorServlet", urlPatterns = {"/EditorServlet"})
public class EditorServlet extends HttpServlet {
public class EditorServlet extends HttpServlet
{
protected void processRequest(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException
{
String fileName = "";
@ -87,17 +87,20 @@ public class EditorServlet extends HttpServlet {
}
@Override
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException
{
processRequest(request, response);
}
@Override
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException
{
processRequest(request, response);
}
@Override
public String getServletInfo() {
return "Short description";
public String getServletInfo()
{
return "Editor page";
}
}

View File

@ -38,61 +38,62 @@ import javax.net.ssl.X509TrustManager;
import javax.servlet.ServletContextEvent;
import javax.servlet.ServletContextListener;
public class GlobalServletContextListener implements ServletContextListener{
@Override
public void contextDestroyed(ServletContextEvent arg0) {
System.out.println("ServletContextListener destroyed");
}
public class GlobalServletContextListener implements ServletContextListener
{
@Override
public void contextDestroyed(ServletContextEvent arg0)
{
System.out.println("ServletContextListener destroyed");
}
@Override
public void contextInitialized(ServletContextEvent arg0) {
TrustManager[] trustAllCerts = new TrustManager[] {
new X509TrustManager() {
@Override
public java.security.cert.X509Certificate[] getAcceptedIssuers()
{
return null;
}
@Override
public void checkClientTrusted(X509Certificate[] certs, String authType)
{
}
@Override
public void checkServerTrusted(X509Certificate[] certs, String authType)
{
}
}
};
SSLContext sc;
try
@Override
public void contextInitialized(ServletContextEvent arg0)
{
TrustManager[] trustAllCerts = new TrustManager[]
{
new X509TrustManager()
{
sc = SSLContext.getInstance("SSL");
sc.init(null, trustAllCerts, new java.security.SecureRandom());
HttpsURLConnection.setDefaultSSLSocketFactory(sc.getSocketFactory());
}
catch (NoSuchAlgorithmException | KeyManagementException ex)
{
}
HostnameVerifier allHostsValid = new HostnameVerifier() {
@Override
public boolean verify(String hostname, SSLSession session) {
return true;
public java.security.cert.X509Certificate[] getAcceptedIssuers()
{
return null;
}
};
HttpsURLConnection.setDefaultHostnameVerifier(allHostsValid);
@Override
public void checkClientTrusted(X509Certificate[] certs, String authType)
{
}
System.out.println("ServletContextListener started");
}
@Override
public void checkServerTrusted(X509Certificate[] certs, String authType)
{
}
}
};
SSLContext sc;
try
{
sc = SSLContext.getInstance("SSL");
sc.init(null, trustAllCerts, new java.security.SecureRandom());
HttpsURLConnection.setDefaultSSLSocketFactory(sc.getSocketFactory());
}
catch (NoSuchAlgorithmException | KeyManagementException ex)
{
}
HostnameVerifier allHostsValid = new HostnameVerifier()
{
@Override
public boolean verify(String hostname, SSLSession session)
{
return true;
}
};
HttpsURLConnection.setDefaultHostnameVerifier(allHostsValid);
System.out.println("ServletContextListener started");
}
}

View File

@ -50,10 +50,10 @@ import org.json.simple.parser.JSONParser;
@WebServlet(name = "IndexServlet", urlPatterns = {"/IndexServlet"})
@MultipartConfig
public class IndexServlet extends HttpServlet {
protected void processRequest(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
public class IndexServlet extends HttpServlet
{
protected void processRequest(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException
{
String action = request.getParameter("type");
if(action == null)
@ -65,7 +65,8 @@ public class IndexServlet extends HttpServlet {
DocumentManager.Init(request, response);
PrintWriter writer = response.getWriter();
switch (action.toLowerCase()) {
switch (action.toLowerCase())
{
case "upload":
Upload(request, response, writer);
break;
@ -76,11 +77,11 @@ public class IndexServlet extends HttpServlet {
Track(request, response, writer);
break;
}
}
private static void Upload(HttpServletRequest request, HttpServletResponse response, PrintWriter writer) {
private static void Upload(HttpServletRequest request, HttpServletResponse response, PrintWriter writer)
{
response.setContentType("text/plain");
try
@ -88,20 +89,24 @@ public class IndexServlet extends HttpServlet {
Part httpPostedFile = request.getPart("file");
String fileName = "";
for (String content : httpPostedFile.getHeader("content-disposition").split(";")) {
if (content.trim().startsWith("filename")) {
for (String content : httpPostedFile.getHeader("content-disposition").split(";"))
{
if (content.trim().startsWith("filename"))
{
fileName = content.substring(content.indexOf('=') + 1).trim().replace("\"", "");
}
}
long curSize = httpPostedFile.getSize();
if (DocumentManager.GetMaxFileSize() < curSize || curSize <= 0) {
if (DocumentManager.GetMaxFileSize() < curSize || curSize <= 0)
{
writer.write("{ \"error\": \"File size is incorrect\"}");
return;
}
String curExt = FileUtility.GetFileExtension(fileName);
if (!DocumentManager.GetFileExts().contains(curExt)) {
if (!DocumentManager.GetFileExts().contains(curExt))
{
writer.write("{ \"error\": \"File type is not supported\"}");
return;
}
@ -113,10 +118,12 @@ public class IndexServlet extends HttpServlet {
File file = new File(fileStoragePath);
try (FileOutputStream out = new FileOutputStream(file)) {
try (FileOutputStream out = new FileOutputStream(file))
{
int read;
final byte[] bytes = new byte[1024];
while ((read = fileStream.read(bytes)) != -1) {
while ((read = fileStream.read(bytes)) != -1)
{
out.write(bytes, 0, read);
}
@ -162,15 +169,18 @@ public class IndexServlet extends HttpServlet {
java.net.HttpURLConnection connection = (java.net.HttpURLConnection) url.openConnection();
InputStream stream = connection.getInputStream();
if (stream == null) {
if (stream == null)
{
throw new Exception("Stream is null");
}
File convertedFile = new File(DocumentManager.StoragePath(correctName, null));
try (FileOutputStream out = new FileOutputStream(convertedFile)) {
try (FileOutputStream out = new FileOutputStream(convertedFile))
{
int read;
final byte[] bytes = new byte[1024];
while ((read = stream.read(bytes)) != -1) {
while ((read = stream.read(bytes)) != -1)
{
out.write(bytes, 0, read);
}
@ -195,7 +205,8 @@ public class IndexServlet extends HttpServlet {
}
}
private static void Track(HttpServletRequest request, HttpServletResponse response, PrintWriter writer) {
private static void Track(HttpServletRequest request, HttpServletResponse response, PrintWriter writer)
{
String userAddress = request.getParameter("userAddress");
String fileName = request.getParameter("fileName");
@ -248,15 +259,18 @@ public class IndexServlet extends HttpServlet {
java.net.HttpURLConnection connection = (java.net.HttpURLConnection) url.openConnection();
InputStream stream = connection.getInputStream();
if (stream == null) {
if (stream == null)
{
throw new Exception("Stream is null");
}
File savedFile = new File(storagePath);
try (FileOutputStream out = new FileOutputStream(savedFile)) {
try (FileOutputStream out = new FileOutputStream(savedFile))
{
int read;
final byte[] bytes = new byte[1024];
while ((read = stream.read(bytes)) != -1) {
while ((read = stream.read(bytes)) != -1)
{
out.write(bytes, 0, read);
}
@ -276,20 +290,21 @@ public class IndexServlet extends HttpServlet {
}
@Override
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException
{
processRequest(request, response);
}
@Override
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException
{
processRequest(request, response);
}
@Override
public String getServletInfo() {
return "Short description";
public String getServletInfo()
{
return "Handler";
}
}

View File

@ -27,7 +27,8 @@
package entities;
public enum FileType {
public enum FileType
{
Text,
Spreadsheet,
Presentation

View File

@ -30,8 +30,8 @@ package helpers;
import java.io.InputStream;
import java.util.Properties;
public class ConfigManager {
public class ConfigManager
{
private static Properties properties;
static
@ -53,7 +53,8 @@ public class ConfigManager {
}
}
public static String GetProperty(String name){
public static String GetProperty(String name)
{
if(properties == null)
return "";

View File

@ -44,7 +44,8 @@ public class DocumentManager
{
private static HttpServletRequest request;
public static void Init(HttpServletRequest req, HttpServletResponse resp){
public static void Init(HttpServletRequest req, HttpServletResponse resp)
{
request = req;
}
@ -157,25 +158,20 @@ public class DocumentManager
{
String demoName = "sample." + fileExt;
String fileName = GetCorrectName(demoName);
try
InputStream stream = Thread.currentThread().getContextClassLoader().getResourceAsStream(demoName);
File file = new File(StoragePath(fileName, null));
try (FileOutputStream out = new FileOutputStream(file))
{
InputStream stream = Thread.currentThread().getContextClassLoader().getResourceAsStream(demoName);
File file = new File(StoragePath(fileName, null));
try (FileOutputStream out = new FileOutputStream(file)) {
int read;
final byte[] bytes = new byte[1024];
while ((read = stream.read(bytes)) != -1) {
out.write(bytes, 0, read);
}
out.flush();
int read;
final byte[] bytes = new byte[1024];
while ((read = stream.read(bytes)) != -1)
{
out.write(bytes, 0, read);
}
}
catch (Exception ex)
{
throw ex;
out.flush();
}
return fileName;
@ -192,12 +188,10 @@ public class DocumentManager
String filePath = serverPath + "/" + storagePath + "/" + hostAddress + "/" + URLEncoder.encode(fileName, java.nio.charset.StandardCharsets.UTF_8.toString());
return filePath;
} catch (UnsupportedEncodingException e) {
throw new AssertionError("UTF-8 is unknown");
}
catch (Exception ex)
catch (UnsupportedEncodingException e)
{
throw ex;
throw new AssertionError("UTF-8 is unknown");
}
}
@ -210,11 +204,14 @@ public class DocumentManager
{
String serverPath = request.getScheme() + "://" + request.getServerName() + ":" + request.getServerPort() + request.getContextPath();
String hostAddress = CurUserHostAddress(null);
try {
try
{
String query = "?type=track&fileName=" + URLEncoder.encode(fileName, java.nio.charset.StandardCharsets.UTF_8.toString()) + "&userAddress=" + URLEncoder.encode(hostAddress, java.nio.charset.StandardCharsets.UTF_8.toString());
return serverPath + "/IndexServlet" + query;
} catch (UnsupportedEncodingException e) {
}
catch (UnsupportedEncodingException e)
{
throw new AssertionError("UTF-8 is unknown");
}
}

View File

@ -37,23 +37,16 @@ import java.net.URLEncoder;
import java.text.MessageFormat;
import java.util.UUID;
import java.util.concurrent.TimeoutException;
import javax.xml.parsers.DocumentBuilder;
import javax.xml.parsers.DocumentBuilderFactory;
import javax.xml.parsers.ParserConfigurationException;
import org.w3c.dom.Document;
import org.w3c.dom.Element;
import org.w3c.dom.NodeList;
import org.xml.sax.InputSource;
import org.xml.sax.SAXException;
import org.json.simple.JSONObject;
import org.json.simple.parser.JSONParser;
import org.json.simple.parser.ParseException;
public class ServiceConverter
{
private static int ConvertTimeout = 120000;
private static final String DocumentConverterUrl = ConfigManager.GetProperty("files.docservice.url.converter");
private static final String DocumentStorageUrl = ConfigManager.GetProperty("files.docservice.url.storage");
private static final MessageFormat ConvertParams = new MessageFormat("?url={0}&outputtype={1}&filetype={2}&title={3}&key={4}");
private static final int MaxTry = 3;
static
{
@ -71,113 +64,14 @@ public class ServiceConverter
}
public static String GetConvertedUri(String documentUri, String fromExtension, String toExtension, String documentRevisionId, Boolean isAsync) throws Exception
{
String convertedDocumentUri = null;
String xml = SendRequestToConvertService(documentUri, fromExtension, toExtension, documentRevisionId, isAsync);
Document document = ConvertStringToXmlDocument(xml);
Element responceFromConvertService = document.getDocumentElement();
if (responceFromConvertService == null)
throw new Exception("Invalid answer format");
NodeList errorElement = responceFromConvertService.getElementsByTagName("Error");
if (errorElement != null && errorElement.getLength() > 0)
ProcessConvertServiceResponceError(Integer.parseInt(errorElement.item(0).getTextContent()));
NodeList endConvertNode = responceFromConvertService.getElementsByTagName("EndConvert");
if (endConvertNode == null || endConvertNode.getLength() == 0)
throw new Exception("EndConvert node is null");
Boolean isEndConvert = Boolean.parseBoolean(endConvertNode.item(0).getTextContent());
NodeList percentNode = responceFromConvertService.getElementsByTagName("Percent");
if (percentNode == null || percentNode.getLength() == 0)
throw new Exception("Percent node is null");
Integer percent = Integer.parseInt(percentNode.item(0).getTextContent());
if (isEndConvert)
{
NodeList fileUrlNode = responceFromConvertService.getElementsByTagName("FileUrl");
if (fileUrlNode == null || fileUrlNode.getLength() == 0)
throw new Exception("FileUrl node is null");
convertedDocumentUri = fileUrlNode.item(0).getTextContent();
percent = 100;
}
else
{
percent = percent >= 100 ? 99 : percent;
}
return percent >= 100 ? convertedDocumentUri : "";
}
public static String GetExternalUri(InputStream fileStream, long contentLength, String contentType, String documentRevisionId) throws IOException, Exception
{
Object[] args = {"", "", "", "", documentRevisionId};
String urlTostorage = DocumentStorageUrl + ConvertParams.format(args);
URL url = new URL(urlTostorage);
java.net.HttpURLConnection connection = (java.net.HttpURLConnection) url.openConnection();
connection.setDoOutput(true);
connection.setDoInput(true);
connection.setInstanceFollowRedirects(false);
connection.setRequestMethod("POST");
connection.setRequestProperty("Content-Type", contentType == null ? "application/octet-stream" : contentType);
connection.setRequestProperty("charset", "utf-8");
connection.setRequestProperty("Content-Length", Long.toString(contentLength));
connection.setUseCaches (false);
try (DataOutputStream dataOutputStream = new DataOutputStream(connection.getOutputStream())) {
int read;
final byte[] bytes = new byte[1024];
while ((read = fileStream.read(bytes)) != -1) {
dataOutputStream.write(bytes, 0, read);
}
dataOutputStream.flush();
}
InputStream stream = connection.getInputStream();
if (stream == null)
{
throw new Exception("Could not get an answer");
}
String xml = ConvertStreamToString(stream);
connection.disconnect();
String res = GetResponseUri(xml);
return res;
}
public static String GenerateRevisionId(String expectedKey)
{
if (expectedKey.length() > 20)
expectedKey = Integer.toString(expectedKey.hashCode());
String key = expectedKey.replace("[^0-9-.a-zA-Z_=]", "_");
return key.substring(0, Math.min(key.length(), 20));
}
private static String SendRequestToConvertService(String documentUri, String fromExtension, String toExtension, String documentRevisionId, Boolean isAsync) throws Exception
{
fromExtension = fromExtension == null || fromExtension.isEmpty() ? FileUtility.GetFileExtension(documentUri) : fromExtension;
String title = FileUtility.GetFileName(documentUri);
title = title == null || title.isEmpty() ? UUID.randomUUID().toString() : title;
documentRevisionId = documentRevisionId == null || documentRevisionId.isEmpty() ? documentUri : documentRevisionId;
documentRevisionId = GenerateRevisionId(documentRevisionId);
Object[] args = {
@ -187,7 +81,7 @@ public class ServiceConverter
title,
documentRevisionId
};
String urlToConverter = DocumentConverterUrl + ConvertParams.format(args);
if (isAsync)
@ -195,38 +89,29 @@ public class ServiceConverter
URL url = new URL(urlToConverter);
java.net.HttpURLConnection connection = (java.net.HttpURLConnection) url.openConnection();
connection.setRequestProperty("Accept", "application/json");
connection.setConnectTimeout(ConvertTimeout);
InputStream stream = null;
int countTry = 0;
while (countTry < MaxTry)
{
try
{
countTry++;
stream = connection.getInputStream();
break;
}
catch (Exception ex)
{
if(!(ex instanceof TimeoutException))
throw new Exception("Bad Request");
}
}
if (countTry == MaxTry)
{
throw new Exception("Timeout");
}
InputStream stream = connection.getInputStream();
if (stream == null)
throw new Exception("Could not get an answer");
String xml = ConvertStreamToString(stream);
String jsonString = ConvertStreamToString(stream);
connection.disconnect();
return xml;
return GetResponseUri(jsonString);
}
public static String GenerateRevisionId(String expectedKey)
{
if (expectedKey.length() > 20)
expectedKey = Integer.toString(expectedKey.hashCode());
String key = expectedKey.replace("[^0-9-.a-zA-Z_=]", "_");
return key.substring(0, Math.min(key.length(), 20));
}
private static void ProcessConvertServiceResponceError(int errorCode) throws Exception
@ -270,46 +155,31 @@ public class ServiceConverter
throw new Exception(errorMessage);
}
private static String GetResponseUri(String xml) throws Exception
private static String GetResponseUri(String jsonString) throws Exception
{
Document document = ConvertStringToXmlDocument(xml);
Element responceFromConvertService = document.getDocumentElement();
if (responceFromConvertService == null)
throw new Exception("Invalid answer format");
JSONObject jsonObj = ConvertStringToJSON(jsonString);
NodeList errorElement = responceFromConvertService.getElementsByTagName("Error");
if (errorElement != null && errorElement.getLength() > 0)
ProcessConvertServiceResponceError(Integer.parseInt(errorElement.item(0).getTextContent()));
String error = (String) jsonObj.get("error");
if (error != null && error != "")
ProcessConvertServiceResponceError(Integer.parseInt(error));
NodeList endConvert = responceFromConvertService.getElementsByTagName("EndConvert");
if (endConvert == null || endConvert.getLength() == 0)
throw new Exception("Invalid answer format");
Boolean isEndConvert = Boolean.parseBoolean(endConvert.item(0).getTextContent());
Boolean isEndConvert = (Boolean) jsonObj.get("endConvert");
int resultPercent = 0;
Long resultPercent = 0l;
String responseUri = null;
if (isEndConvert)
{
NodeList fileUrl = responceFromConvertService.getElementsByTagName("FileUrl");
if (fileUrl == null || endConvert.getLength() == 0)
throw new Exception("Invalid answer format");
resultPercent = 100;
responseUri = fileUrl.item(0).getTextContent();
resultPercent = 100l;
responseUri = (String) jsonObj.get("fileUrl");
}
else
{
NodeList percent = responceFromConvertService.getElementsByTagName("Percent");
if (percent != null && percent.getLength() > 0)
resultPercent = Integer.parseInt(percent.item(0).getTextContent());
resultPercent = resultPercent >= 100 ? 99 : resultPercent;
resultPercent = (Long) jsonObj.get("percent");
resultPercent = resultPercent >= 100l ? 99l : resultPercent;
}
return resultPercent >= 100 ? responseUri : "";
return resultPercent >= 100l ? responseUri : "";
}
private static String ConvertStreamToString(InputStream stream) throws IOException
@ -319,24 +189,23 @@ public class ServiceConverter
BufferedReader bufferedReader = new BufferedReader(inputStreamReader);
String line = bufferedReader.readLine();
while(line != null) {
while(line != null)
{
stringBuilder.append(line);
line =bufferedReader.readLine();
line = bufferedReader.readLine();
}
String result = stringBuilder.toString();
return result;
}
private static Document ConvertStringToXmlDocument(String xml) throws IOException, ParserConfigurationException, SAXException
private static JSONObject ConvertStringToJSON(String jsonString) throws ParseException
{
DocumentBuilderFactory documentBuildFactory = DocumentBuilderFactory.newInstance();
DocumentBuilder doccumentBuilder = documentBuildFactory.newDocumentBuilder();
InputStream inputStream = new ByteArrayInputStream(xml.getBytes("utf-8"));
InputSource inputSource = new InputSource(inputStream);
Document document = doccumentBuilder.parse(inputSource);
return document;
JSONParser parser = new JSONParser();
Object obj = parser.parse(jsonString);
JSONObject jsonObj = (JSONObject) obj;
return jsonObj;
}
}

View File

@ -6,7 +6,6 @@ files.docservice.edited-docs=.docx|.xlsx|.csv|.pptx|.ppsx|.txt
files.docservice.convert-docs=.docm|.dotx|.dotm|.dot|.doc|.odt|.fodt|.xlsm|.xltx|.xltm|.xlt|.xls|.ods|.fods|.pptm|.ppt|.ppsm|.pps|.potx|.potm|.pot|.odp|.fodp|.rtf|.mht|.html|.htm|.epub
files.docservice.timeout=120000
files.docservice.url.storage=https://doc.onlyoffice.com/FileUploader.ashx
files.docservice.url.converter=https://doc.onlyoffice.com/ConvertService.ashx
files.docservice.url.tempstorage=https://doc.onlyoffice.com/ResourceService.ashx
files.docservice.url.api=https://doc.onlyoffice.com/web-apps/apps/api/documents/api.js

View File

@ -40,6 +40,7 @@
<link rel="icon" href="favicon.ico" type="image/x-icon" />
<link rel="stylesheet" type="text/css" href="css/editor.css" />
<% DocumentManager.Init(request, response); %>
<% FileModel Model = (FileModel)request.getAttribute("file"); %>
<script type="text/javascript" src="${docserviceApiUrl}"></script>

View File

@ -14,7 +14,6 @@
"port": 3000,
"siteUrl": "https://doc.onlyoffice.com/",
"commandUrl": "coauthoring/CommandService.ashx",
"storageUrl": "FileUploader.ashx",
"converterUrl": "ConvertService.ashx",
"tempStorageUrl": "ResourceService.ashx",
"apiUrl": "web-apps/apps/api/documents/api.js",

View File

@ -27,7 +27,6 @@
var path = require("path");
var urlModule = require("url");
var urllib = require("urllib");
var xml2js = require("xml2js");
var jwt = require("jsonwebtoken");
var jwa = require("jwa");
var fileUtility = require("./fileUtility");
@ -77,7 +76,8 @@ documentService.getConvertedUri = function (documentUri, fromExtension, toExtens
var uri = siteUrl + configServer.get('converterUrl');
var headers = {
'Content-Type': 'application/json'
'Content-Type': 'application/json',
"Accept": "application/json"
};
if (cfgSignatureEnable && cfgSignatureUseForRequest) {
@ -93,38 +93,6 @@ documentService.getConvertedUri = function (documentUri, fromExtension, toExtens
callback);
};
documentService.getExternalUri = function (fileStream, contentLength, contentType, documentRevisionId, callback) {
documentRevisionId = documentService.generateRevisionId(documentRevisionId);
var urlTostorage = siteUrl + configServer.get('storageUrl') + "?key=" + documentRevisionId;
var headers = {
"Content-Type": contentType == null ? "application/octet-stream" : contentType,
"Content-Length": contentLength.toString(),
"charset": "utf-8"
};
if (cfgSignatureEnable && cfgSignatureUseForRequest) {
const hmac = jwa(cfgSignatureSecretAlgorithmRequest);
var payloadhash = hmac.sign(fileStream, cfgSignatureSecret);
headers[cfgSignatureAuthorizationHeader] = cfgSignatureAuthorizationHeaderPrefix + this.fillJwtByUrl(urlTostorage, undefined, undefined, payloadhash);
}
urllib.request(urlTostorage,
{
method: "POST",
headers: headers,
data: fileStream
},
function (err, data) {
if (err) {
callback();
return;
}
var res = documentService.getResponseUri(data);
callback(res.value);
});
};
documentService.generateRevisionId = function (expectedKey) {
if (expectedKey.length > 20) {
expectedKey = expectedKey.hashCode().toString();
@ -177,33 +145,22 @@ documentService.processConvertServiceResponceError = function (errorCode) {
throw { message: errorMessage };
};
documentService.getResponseUri = function (xml) {
var json = documentService.convertXmlStringToJson(xml);
documentService.getResponseUri = function (json) {
var fileResult = JSON.parse(json);
if (!json.FileResult)
throw { message: "FileResult node is null" };
if (fileResult.error)
documentService.processConvertServiceResponceError(parseInt(fileResult.error));
var fileResult = json.FileResult;
var isEndConvert = fileResult.endConvert
if (fileResult.Error)
documentService.processConvertServiceResponceError(parseInt(fileResult.Error[0]));
if (!fileResult.EndConvert)
throw { message: "EndConvert node is null" };
var isEndConvert = fileResult.EndConvert[0].toLowerCase() === "true";
if (!fileResult.Percent)
throw { message: "Percent node is null" };
var percent = parseInt(fileResult.Percent[0]);
var percent = parseInt(fileResult.percent);
var uri = null;
if (isEndConvert) {
if (!fileResult.FileUrl)
throw { message: "FileUrl node is null" };
if (!fileResult.fileUrl)
throw { message: "FileUrl is null" };
uri = fileResult.FileUrl[0];
uri = fileResult.fileUrl;
percent = 100;
} else {
percent = percent >= 100 ? 99 : percent;
@ -215,16 +172,6 @@ documentService.getResponseUri = function (xml) {
};
};
documentService.convertXmlStringToJson = function (xml) {
var res;
xml2js.parseString(xml, function (err, result) {
res = result;
});
return res;
};
documentService.commandRequest = function (method, documentRevisionId, callback) {
documentRevisionId = documentService.generateRevisionId(documentRevisionId);
var params = {

View File

@ -26,8 +26,7 @@
"mime": "^1.3.4",
"serve-favicon": "~2.3.0",
"sync-request": "^4.0.1",
"urllib": "^2.20.0",
"xml2js": "~0.4.17"
"urllib": "^2.20.0"
},
"license": "MIT",
"repository": {

View File

@ -12,7 +12,6 @@ $GLOBALS['DOC_SERV_CONVERT'] = array(".docm", ".doc", ".dotx", ".dotm", ".dot",
$GLOBALS['DOC_SERV_TIMEOUT'] = "120000";
$GLOBALS['DOC_SERV_STORAGE_URL'] = "https://doc.onlyoffice.com/FileUploader.ashx";
$GLOBALS['DOC_SERV_CONVERTER_URL'] = "https://doc.onlyoffice.com/ConvertService.ashx";
$GLOBALS['DOC_SERV_API_URL'] = "https://doc.onlyoffice.com/web-apps/apps/api/documents/api.js";
@ -38,8 +37,5 @@ $GLOBALS['ExtsDocument'] = array(".doc", ".docx", ".docm",
".html", ".htm", ".mht",
".pdf", ".djvu", ".fb2", ".epub", ".xps");
if ( !defined('ServiceConverterMaxTry') )
define( 'ServiceConverterMaxTry', 3);
?>

View File

@ -31,56 +31,6 @@
require_once( dirname(__FILE__) . '/config.php' );
function GetExternalFileUri($local_uri) {
$externalUri = '';
try
{
$documentRevisionId = GenerateRevisionId($local_uri);
if (($fileContents = file_get_contents(str_replace(" ","%20", $local_uri))) === FALSE) {
throw new Exception("Bad Request");
} else {
$contentType = mime_content_type($local_uri);
$urlToService = generateUrlToStorage('', '', '', '', $documentRevisionId);
$opts = array('http' => array(
'method' => 'POST',
'header' => "User-Agent: " . $_SERVER['HTTP_USER_AGENT'] . "\r\n" .
"Content-Type: " . $contentType . "\r\n" .
"Content-Length: " . strlen($fileContents) . "\r\n",
'content' => $fileContents,
'timeout' => $GLOBALS['DOC_SERV_TIMEOUT']
)
);
if (substr($urlToService, 0, strlen("https")) === "https") {
$opts['ssl'] = array( 'verify_peer' => FALSE );
}
$context = stream_context_create($opts);
if (($response_data = file_get_contents($urlToService, FALSE, $context)) === FALSE) {
throw new Exception ("Could not get an answer");
} else {
sendlog("GetExternalUri response_data:" . PHP_EOL . $response_data, "logs/common.log");
GetResponseUri($response_data, $externalUri);
}
sendlog("GetExternalFileUri. externalUri = " . $externalUri, "logs/common.log");
return $externalUri . "";
}
}
catch (Exception $e)
{
sendlog("GetExternalFileUri Exception: " . $e->getMessage(), "logs/common.log");
}
return $local_uri;
}
function DoUpload($fileUri) {
$_fileName = GetCorrectName($fileUri);
@ -101,18 +51,6 @@ function DoUpload($fileUri) {
}
function generateUrlToStorage($document_uri, $from_extension, $to_extension, $title, $document_revision_id) {
return $GLOBALS['DOC_SERV_STORAGE_URL'] . "?" . http_build_query(
array(
"url" => $document_uri,
"outputtype" => trim($to_extension,'.'),
"filetype" => trim($from_extension, '.'),
"title" => $title,
"key" => $document_revision_id));
}
/**
* Generate an error code table
*
@ -185,7 +123,7 @@ function GenerateRevisionId($expected_key) {
* @param string $document_revision_id Key for caching on service
* @param bool $is_async Perform conversions asynchronously
*
* @return Xml document request result of conversion
* @return Document request result of conversion
*/
function SendRequestToConvertService($document_uri, $from_extension, $to_extension, $document_revision_id, $is_async) {
if (empty($from_extension))
@ -218,13 +156,11 @@ function SendRequestToConvertService($document_uri, $from_extension, $to_extensi
)
);
$response_xml_data;
$countTry = 0;
$opts = array('http' => array(
'method' => 'POST',
'timeout' => $GLOBALS['DOC_SERV_TIMEOUT'],
'header'=> "Content-type: application/json\r\n",
'header'=> "Content-type: application/json\r\n" .
"Accept: application/json\r\n",
'content' => $data
)
);
@ -234,30 +170,7 @@ function SendRequestToConvertService($document_uri, $from_extension, $to_extensi
}
$context = stream_context_create($opts);
while ($countTry < ServiceConverterMaxTry)
{
$countTry = $countTry + 1;
$response_xml_data = file_get_contents($urlToConverter, FALSE, $context);
if ($response_xml_data !== false) { break; }
}
if ($countTry == ServiceConverterMaxTry)
{
throw new Exception ("Bad Request or timeout error");
}
libxml_use_internal_errors(true);
if (!function_exists('simplexml_load_file')) {
throw new Exception("Server can't read xml");
}
$response_data = simplexml_load_string($response_xml_data);
if (!$response_data) {
$exc = "Bad Response. Errors: ";
foreach(libxml_get_errors() as $error) {
$exc = $exc . "\t" . $error->message;
}
throw new Exception ($exc);
}
$response_data = file_get_contents($urlToConverter, FALSE, $context);
return $response_data;
}
@ -304,39 +217,28 @@ function GetConvertedUri($document_uri, $from_extension, $to_extension, $documen
/**
* Processing document received from the editing service.
*
* @param string $x_document_response The resulting xml from editing service
* @param string $document_response The result from editing service
* @param string $response_uri Uri to the converted document
*
* @return The percentage of completion of conversion
*/
function GetResponseUri($x_document_response, &$response_uri) {
function GetResponseUri($document_response, &$response_uri) {
$response_uri = "";
$resultPercent = 0;
libxml_use_internal_errors(true);
if (!function_exists('simplexml_load_file')) {
throw new Exception("Server can't read xml");
}
$data = simplexml_load_string($x_document_response);
if (!$data) {
$errs = "Invalid answer format. Errors: ";
foreach(libxml_get_errors() as $error) {
$errs = $errs . '\t' . $error->message;
}
throw new Exception ($errs);
if (!$document_response) {
$errs = "Invalid answer format";
}
$errorElement = $data->Error;
if ($errorElement != NULL && $errorElement != "") ProcessConvServResponceError($data->Error);
$errorElement = $document_response->Error;
if ($errorElement != NULL && $errorElement != "") ProcessConvServResponceError($document_response->Error);
$endConvert = $data->EndConvert;
$endConvert = $document_response->EndConvert;
if ($endConvert != NULL && $endConvert == "") throw new Exception("Invalid answer format");
if ($endConvert != NULL && strtolower($endConvert) == true)
{
$fileUrl = $data->FileUrl;
$fileUrl = $document_response->FileUrl;
if ($fileUrl == NULL || $fileUrl == "") throw new Exception("Invalid answer format");
$response_uri = $fileUrl;
@ -344,7 +246,7 @@ function GetResponseUri($x_document_response, &$response_uri) {
}
else
{
$percent = $data->Percent;
$percent = $document_response->Percent;
if ($percent != NULL && $percent != "")
$resultPercent = $percent;

View File

@ -36,51 +36,51 @@ GEM
minitest (~> 5.1)
thread_safe (~> 0.3, >= 0.3.4)
tzinfo (~> 1.1)
arel (6.0.0)
arel (6.0.4)
binding_of_caller (0.7.2)
debug_inspector (>= 0.0.1)
builder (3.2.2)
byebug (3.5.1)
columnize (~> 0.8)
debugger-linecache (~> 1.2)
slop (~> 3.6)
coffee-rails (4.1.0)
builder (3.2.3)
byebug (9.1.0)
coffee-rails (4.1.1)
coffee-script (>= 2.2.0)
railties (>= 4.0.0, < 5.0)
coffee-script (2.3.0)
railties (>= 4.0.0, < 5.1.x)
coffee-script (2.4.1)
coffee-script-source
execjs
coffee-script-source (1.8.0)
columnize (0.9.0)
debug_inspector (0.0.2)
debugger-linecache (1.2.0)
coffee-script-source (1.12.2)
concurrent-ruby (1.0.5)
crass (1.0.2)
debug_inspector (0.0.3)
erubis (2.7.0)
execjs (2.2.2)
globalid (0.3.0)
activesupport (>= 4.1.0)
hike (1.2.3)
i18n (0.7.0)
jbuilder (2.2.6)
activesupport (>= 3.0.0, < 5)
multi_json (~> 1.2)
jquery-rails (4.0.3)
rails-dom-testing (~> 1.0)
execjs (2.7.0)
ffi (1.9.18)
globalid (0.4.0)
activesupport (>= 4.2.0)
i18n (0.8.6)
jbuilder (2.7.0)
activesupport (>= 4.2.0)
multi_json (>= 1.2)
jquery-rails (4.3.1)
rails-dom-testing (>= 1, < 3)
railties (>= 4.2.0)
thor (>= 0.14, < 2.0)
json (1.8.2)
loofah (2.0.1)
json (1.8.6)
loofah (2.1.1)
crass (~> 1.0.2)
nokogiri (>= 1.5.9)
macaddr (1.7.1)
systemu (~> 2.6.2)
mail (2.6.3)
mime-types (>= 1.16, < 3)
mime-types (2.4.3)
mini_portile (0.6.2)
minitest (5.5.1)
multi_json (1.10.1)
nokogiri (1.6.6.1-x86-mingw32)
mini_portile (~> 0.6.0)
rack (1.6.0)
mail (2.6.6)
mime-types (>= 1.16, < 4)
mime-types (3.1)
mime-types-data (~> 3.2015)
mime-types-data (3.2016.0521)
mini_portile2 (2.3.0)
minitest (5.10.3)
multi_json (1.12.2)
nokogiri (1.8.1)
mini_portile2 (~> 2.3.0)
rack (1.6.8)
rack-test (0.6.3)
rack (>= 1.0)
rails (4.2.0)
@ -96,64 +96,65 @@ GEM
sprockets-rails
rails-deprecated_sanitizer (1.0.3)
activesupport (>= 4.2.0.alpha)
rails-dom-testing (1.0.5)
rails-dom-testing (1.0.8)
activesupport (>= 4.2.0.beta, < 5.0)
nokogiri (~> 1.6.0)
nokogiri (~> 1.6)
rails-deprecated_sanitizer (>= 1.0.1)
rails-html-sanitizer (1.0.1)
rails-html-sanitizer (1.0.3)
loofah (~> 2.0)
railties (4.2.0)
actionpack (= 4.2.0)
activesupport (= 4.2.0)
rake (>= 0.8.7)
thor (>= 0.18.1, < 2.0)
rake (10.4.2)
rdoc (4.2.0)
json (~> 1.4)
sass (3.4.10)
sass-rails (5.0.1)
railties (>= 4.0.0, < 5.0)
rake (12.1.0)
rb-fsevent (0.10.2)
rb-inotify (0.9.10)
ffi (>= 0.5.0, < 2)
rdoc (4.3.0)
sass (3.5.2)
sass-listen (~> 4.0.0)
sass-listen (4.0.0)
rb-fsevent (~> 0.9, >= 0.9.4)
rb-inotify (~> 0.9, >= 0.9.7)
sass-rails (5.0.6)
railties (>= 4.0.0, < 6)
sass (~> 3.1)
sprockets (>= 2.8, < 4.0)
sprockets-rails (>= 2.0, < 4.0)
tilt (~> 1.1)
sdoc (0.4.1)
tilt (>= 1.1, < 3)
sdoc (0.4.2)
json (~> 1.7, >= 1.7.7)
rdoc (~> 4.0)
slop (3.6.0)
sprockets (2.12.3)
hike (~> 1.2)
multi_json (~> 1.0)
rack (~> 1.0)
tilt (~> 1.1, != 1.3.0)
sprockets-rails (2.2.4)
actionpack (>= 3.0)
activesupport (>= 3.0)
sprockets (>= 2.8, < 4.0)
sqlite3 (1.3.10-x86-mingw32)
systemu (2.6.4)
thor (0.19.1)
thread_safe (0.3.4)
tilt (1.4.1)
turbolinks (2.5.3)
coffee-rails
tzinfo (1.2.2)
sprockets (3.7.1)
concurrent-ruby (~> 1.0)
rack (> 1, < 3)
sprockets-rails (3.2.1)
actionpack (>= 4.0)
activesupport (>= 4.0)
sprockets (>= 3.0.0)
sqlite3 (1.3.13)
systemu (2.6.5)
thor (0.20.0)
thread_safe (0.3.6)
tilt (2.0.8)
turbolinks (5.0.1)
turbolinks-source (~> 5)
turbolinks-source (5.0.3)
tzinfo (1.2.3)
thread_safe (~> 0.1)
tzinfo-data (1.2014.10)
tzinfo (>= 1.0.0)
uglifier (2.7.0)
execjs (>= 0.3.0)
json (>= 1.8.0)
uuid (2.3.7)
uglifier (3.2.0)
execjs (>= 0.3.0, < 3)
uuid (2.3.8)
macaddr (~> 1.0)
web-console (2.0.0)
activemodel (~> 4.0)
web-console (2.3.0)
activemodel (>= 4.0)
binding_of_caller (>= 0.7.2)
railties (~> 4.0)
railties (>= 4.0)
sprockets-rails (>= 2.0, < 4.0)
PLATFORMS
x86-mingw32
ruby
DEPENDENCIES
byebug
@ -169,3 +170,6 @@ DEPENDENCIES
uglifier (>= 1.3.0)
uuid
web-console (~> 2.0)
BUNDLED WITH
1.15.4

View File

@ -1,3 +1,5 @@
require 'net/http'
class HomeController < ApplicationController
def index
end
@ -117,7 +119,7 @@ class HomeController < ApplicationController
if status == 2 || status == 3 #MustSave, Corrupted
saved = 1
saved = 0
begin
@ -143,12 +145,15 @@ class HomeController < ApplicationController
end
rescue
saved = 0
saved = 1
end
render :text => '{"error":0,"c":"saved","status":' + saved.to_s + '}'
render :text => '{"error":' + saved.to_s + '}'
return
end
render :text => '{"error":0}'
return
end
end

View File

@ -2,82 +2,11 @@ class ServiceConverter
@@convert_timeout = Rails.configuration.timeout
@@document_converter_url = Rails.configuration.urlConverter
@@document_storage_url = Rails.configuration.urlStorage
@@convert_params = '?url=%s&outputtype=%s&filetype=%s&title=%s&key=%s'
@@max_try = 3
class << self
def get_converted_uri(document_uri, from_ext, to_ext, document_revision_id, is_async)
converted_document_uri = nil
responce_from_convert_service = send_request_to_convert_service(document_uri, from_ext, to_ext, document_revision_id, is_async)
file_result = responce_from_convert_service['FileResult']
error_element = file_result['Error']
if error_element != nil
process_convert_service_responce_error(error_element.to_i)
end
is_end_convert = file_result['EndConvert'].downcase == 'true'
percent = file_result['Percent'].to_i
if is_end_convert
converted_document_uri = file_result['FileUrl']
percent = 100
else
percent = percent >= 100 ? 99 : percent;
end
return percent, converted_document_uri
end
def get_external_uri(content, content_length, content_type, document_revision_id)
url_to_storage = @@document_storage_url + (@@convert_params % ['', '', '', '', document_revision_id])
if content_type == nil
content_type = 'application/octet-stream'
end
uri = URI.parse(url_to_storage)
http = Net::HTTP.new(uri.host, uri.port)
if url_to_storage.start_with?('https')
http.use_ssl = true
http.verify_mode = OpenSSL::SSL::VERIFY_NONE
end
req = Net::HTTP::Post.new(uri.request_uri, {'Content-Type' => content_type , 'Content-Length' => content_length.to_s })
req.body = content
res = http.request(req)
data = res.body
if data == nil
raise 'Could not get an answer'
end
document_response = Hash.from_xml(data.gsub('\n', ''))
percent, external_uri = get_response_uri(document_response)
external_uri
end
def generate_revision_id(expected_key)
require 'zlib'
if expected_key.length > 20
expected_key = (Zlib.crc32 expected_key).to_s
end
key = expected_key.gsub(/[^0-9a-zA-Z.=]/, '_')
key[(key.length - [key.length, 20].min)..key.length]
end
def send_request_to_convert_service(document_uri, from_ext, to_ext, document_revision_id, is_async)
from_ext = from_ext == nil ? File.extname(document_uri) : from_ext
@ -95,38 +24,41 @@ class ServiceConverter
end
data = nil
count_try = 0
begin
while count_try < @@max_try
begin
count_try += 1
uri = URI.parse(url_to_converter)
http = Net::HTTP.new(uri.host, uri.port)
uri = URI.parse(url_to_converter)
http = Net::HTTP.new(uri.host, uri.port)
if url_to_converter.start_with?('https')
http.use_ssl = true
http.verify_mode = OpenSSL::SSL::VERIFY_NONE
end
if url_to_converter.start_with?('https')
http.use_ssl = true
http.verify_mode = OpenSSL::SSL::VERIFY_NONE
end
http.read_timeout = @@convert_timeout
req = Net::HTTP::Get.new(uri.request_uri)
res = http.request(req)
data = res.body
break
rescue TimeoutError
#try again
rescue => ex
raise ex.message
end
http.read_timeout = @@convert_timeout
req = Net::HTTP::Get.new(uri.request_uri)
req.add_field("Accept", "application/json")
res = http.request(req)
data = res.body
rescue TimeoutError
#try again
rescue => ex
raise ex.message
end
if count_try == @@max_try && data == nil
raise 'timeout'
json_data = JSON.parse(data)
return get_response_uri(json_data)
end
def generate_revision_id(expected_key)
require 'zlib'
if expected_key.length > 20
expected_key = (Zlib.crc32 expected_key).to_s
end
Hash.from_xml(data.gsub('\n', ''))
key = expected_key.gsub(/[^0-9a-zA-Z.=]/, '_')
key[(key.length - [key.length, 20].min)..key.length]
end
@ -161,30 +93,23 @@ class ServiceConverter
end
def get_response_uri(document_response)
def get_response_uri(json_data)
file_result = document_response['FileResult']
if file_result == nil
raise 'Invalid answer format'
end
file_result = json_data
error_element = file_result['Error']
error_element = file_result['error']
if error_element != nil
process_convert_service_responce_error(error_element.to_i)
end
end_convert_element = file_result['EndConvert']
if end_convert_element == nil
raise 'Invalid answer format'
end
is_end_convert = end_convert_element.downcase == 'true'
is_end_convert = file_result['endConvert']
result_percent = 0
response_uri = ''
if is_end_convert
file_url_element = file_result['FileUrl']
file_url_element = file_result['fileUrl']
if file_url_element == nil
raise 'Invalid answer format'
@ -195,7 +120,7 @@ class ServiceConverter
else
percent_element = file_result['Percent']
percent_element = file_result['percent']
if percent_element != nil
result_percent = percent_element.to_i

View File

@ -39,7 +39,6 @@ module OnlineEditorsExampleRuby
Rails.configuration.editedDocs=".docx|.xlsx|.csv|.pptx|.ppsx|.txt"
Rails.configuration.convertDocs=".docm|.dotx|.dotm|.dot|.doc|.odt|.fodt|.xlsm|.xltx|.xltm|.xlt|.xls|.ods|.fods|.pptm|.ppt|.ppsm|.pps|.potx|.potm|.pot|.odp|.fodp|.rtf|.mht|.html|.htm|.epub"
Rails.configuration.urlStorage="https://doc.onlyoffice.com/FileUploader.ashx"
Rails.configuration.urlConverter="https://doc.onlyoffice.com/ConvertService.ashx"
Rails.configuration.urlApi="https://doc.onlyoffice.com/web-apps/apps/api/documents/api.js"
Rails.configuration.urlPreloader="https://doc.onlyoffice.com/web-apps/apps/api/documents/cache-scripts.html"