###伺服器:WindowsServer 2016 ###Cloudreve 需求方想整一個在小團隊內部使用的網盤系統,最終在千挑萬選之下選中了Cloudreve。 Github地址:https://github.com/cloudreve/Cloudreve 官網地址:https://docs.c ...
伺服器:WindowsServer 2016
Cloudreve
需求方想整一個在小團隊內部使用的網盤系統,最終在千挑萬選之下選中了Cloudreve
。
Cloudreve
是採用go + React
開發的開源網盤系統,優點是部署簡單、界面功能精簡好用、更新比較活躍、文檔詳細,目前在Github
的Star 1.6w+
。
第一步:下載
下載後的文件結構如下圖所示(只有cloudreve.exe
和conf.ini
)
提取碼: dtbb
第二步:修改配置文件
配置文件主要還是配置埠和資料庫,其他詳細配置可以去官網查看
[System]
Debug = false
Mode = master
Listen = :7018
SessionSecret = iA9nowZRHi6s5XSAFEVrNl0njyqZfjmx6mZQGWVBbMQG32SE5dx6pVWPTNTeByTh
HashIDSalt = 5hrN8NvlA7LxdVJCkjlHb5A1Ut2Rph85GqfhwZWvpn2tnGi79uEomziXkHStFMZT
[Database]
Type = mysql
Port = 3306
User = cloudreve
Password = xxxxx
Host = 0.0.0.0
Name = Cloudreve
TablePrefix = cd
Charset = utf8
第三步:啟動
雙擊cloudreve.exe
文件便可以直接啟動,啟動成功的界面如下:
初始化的資料庫表:
真正使用還是建議部署成windows
服務,這時候可以使用上面提到的nssm
工具。
cd
到nssm
工具對應的目錄,輸入:nssm install
便會出現如下界面:
安裝成功後,進到服務管理啟動對應的服務即可。
系統成功啟動後的界面如下:
kkFileView
網盤部署好了,需求方又提了個需求,想要支持線上預覽。
Cloudreve
自帶預覽PDF
和TXT
格式的文件,如果需要預覽Office
相關文件就需要自己對接第三方服務。
目前提供線上預覽的第三方服務有很多,比如微軟大大就提供有免費的文檔線上預覽服務,但前面有說到,網盤系統是在內部使用,這一條只能pass
,第三方不行那就自己做第三方。
於是,萬能的Github
又登場了。
kkFileView
是採用java
開發的文件文檔線上預覽方案,目前在Github
的Star 7.8K+
。
兩個系統都有個優點,就是部署簡單,文檔詳細。
下載下來的文件結構如下,點擊 startup.bat
文件等待片刻後在瀏覽器中打開:http://localhost:8012
便會看到kkFileView
的測試頁面。
如果能看到上圖就表示啟動成功了,大家可以自行在測試頁面上測試相關文件的預覽效果,具體的參數配置可以參考官方文檔,非常詳細。
集成
上面兩步我們已經將網盤服務和線上預覽服務都部署成功了,接下來就是合二為一。
我們先進到Cloudreve
的管理面板 —— 參數設置,有兩個信息我們需要配置。
1、站點URL記得添加埠號
2、圖像與預覽修改Office 文檔預覽服務地址:http://127.0.0.1:8012/onlinePreview?url={$srcB64}
最後點擊保存幾即可。
配置成功後,我們去試試預覽效果,雙擊對應的文件:
誒,怎麼回事,預覽失敗了,而且這個文件類型 lpmatwo4pg1n8bc6
是什麼鬼。
其實這與kkFileView
的預覽機制有關係,kkFileView
預覽是通過下載地址的文件尾碼名去判斷文件類型,但Cloudreve
生成的下載鏈接文件名是一串隨即字元且不包括文件尾碼,所以kkFileView
在處理時不知道對應的文件類型,導致預覽失敗。
知道問題了就有解決辦法:
-
修改
Cloudreve
的下載鏈接生成邏輯,使其帶上文件尾碼 -
修改
kkFileView
的文件類型識別邏輯
因為Cloudreve
是將前後端統一打包成exe
可執行程式,如果修改後還要打包太麻煩,所以放棄解決辦法一(其實這個解決辦法才是相對科學的。)
修改
先去Gitee
或者Github
下載kkFleView
源碼
下載下來後可以根據如下路徑找到對應文件:
file-online-preview-master\server\src\main\java\cn\keking\web\controller\OnlinePreviewController.java
修改思路:根據文件的前n
個起始位元組,也被稱為魔數去判斷對應文件屬於什麼文件類型,但是這個方法也存在缺陷,wps
和office
相關文件的魔數一樣,無法做詳細區分,比如:docx、xlsx、pptx
。
上代碼
因為本人是微軟陣營的,對Java
這塊不熟悉,代碼僅做參考。
1、建立魔數與文件類型的鍵值對映射關係(只列舉了常用的一些,準不准確大家可以自行驗證)
public final static Map<String, String> FILE_TYPE_MAP = new HashMap<String, String>();
static {
FILE_TYPE_MAP.put("ffd8ff", "jpg");
FILE_TYPE_MAP.put("89504e47", "png");
FILE_TYPE_MAP.put("47494638", "gif");
FILE_TYPE_MAP.put("49492a00227105008037", "tif");
FILE_TYPE_MAP.put("424d228c010000000000", "bmp"); // 16色點陣圖(bmp)
FILE_TYPE_MAP.put("424d8240090000000000", "bmp"); // 24位點陣圖(bmp)
FILE_TYPE_MAP.put("424d8e1b030000000000", "bmp"); // 256色點陣圖(bmp)
FILE_TYPE_MAP.put("41433130313500000000", "dwg");
FILE_TYPE_MAP.put("68746d6c3e", "html"); // HTML
FILE_TYPE_MAP.put("48544d4c207b0d0a0942", "css");
FILE_TYPE_MAP.put("696b2e71623d696b2e71", "js");
FILE_TYPE_MAP.put("38425053000100000000", "psd");
FILE_TYPE_MAP.put("255044462d312e", "pdf");
FILE_TYPE_MAP.put("75736167", "txt");
FILE_TYPE_MAP.put("00000020667479706d70", "mp4");
FILE_TYPE_MAP.put("49443303000000002176", "mp3");
FILE_TYPE_MAP.put("41564920", "avi");
FILE_TYPE_MAP.put("3c3f786d6c", "xml");// xml文件
FILE_TYPE_MAP.put("504b03040a000", "office");// office文件
FILE_TYPE_MAP.put("504b030414000", "office");// office文件
FILE_TYPE_MAP.put("d0cf11e0a1b11ae10000", "wps");// WPS文字wps、表格et、演示dps都是一樣的
}
註意:office
和wps
下的相關文件(docx、xlsx、pptx
)的魔數都一樣,所以大家可以預設設置成其中一種即可(我預設的是xlsx
),kkFileView
預覽office
相關文件依賴的是OpenOffice
和LibreOffice
。
2、根據文件流判斷文件類型
/**
* 獲取文件類型
*/
private String getFileType(String fileUrl) throws IOException {
URL newUrl = new URL(fileUrl);
HttpURLConnection conn = (HttpURLConnection) newUrl.openConnection();
InputStream inputStream = conn.getInputStream();
byte[] bytes = IOUtils.toByteArray(inputStream);
StringBuilder stringBuilder = new StringBuilder();
for (int i = 0; i < 20; i++) {
int v = bytes[i] & 0xFF;
String hv = Integer.toHexString(v);
if (hv.length() < 2) {
stringBuilder.append(0);
}
stringBuilder.append(hv);
}
String fileCode = stringBuilder.toString().toLowerCase();
String fileType = null;
Iterator<String> keyIter = FILE_TYPE_MAP.keySet().iterator();
while (keyIter.hasNext()) {
String key = keyIter.next();
if (fileCode.startsWith(key)) {
fileType = FILE_TYPE_MAP.get(key);
break;
}
}
return fileType;
}
3、找到OnlinePreviewController.java
文件的對應方法
主要是在getFileAttribute
方法內加一層尾碼名的處理,為了不影響原方法,所以我們新增一個
/**
* 獲取文件屬性(擴展無尾碼名)
*
* @param url url
* @return 文件屬性
*/
public FileAttribute getFileAttributeEx(String url, String suffix, HttpServletRequest req) {
FileAttribute attribute = new FileAttribute();
FileType type;
String fileName;
String fullFileName = WebUtils.getUrlParameterReg(url, "fullfilename");
if (StringUtils.hasText(fullFileName)) {
fileName = fullFileName;
type = FileType.typeFromFileName(fullFileName);
suffix = KkFileUtils.suffixFromFileName(fullFileName);
} else {
fileName = WebUtils.getFileNameFromURL(url);
String newUrl = url;
int index = newUrl.indexOf("?");
if (index < 0) {
newUrl = newUrl + "." + suffix;
} else {
newUrl = newUrl.substring(0, index) + "." + suffix + newUrl.substring(index, newUrl.length() - 1);
}
type = FileType.typeFromUrl(newUrl);
suffix = WebUtils.suffixFromUrl(newUrl);
}
if (url.contains("?fileKey=")) {
attribute.setSkipDownLoad(true);
}
attribute.setType(type);
attribute.setName(fileName + "." + suffix);
attribute.setSuffix(suffix);
attribute.setUrl(url);
......
return attribute;
}
完事後我們再來看看預覽效果:
原文件:
預覽效果:
註意:kkFileView
對xlsx
和xls
文件的處理預設是以html
形式轉換,其他文件都是以pdf
形式,由於xlsx
和xls
轉pdf
後的效果太過凌亂,權衡利弊下office
和wps
相關的文件全部指定為xlsx
,以html
形式展示。
最後,這個問題其實很早就有人提出過,Cloudreve
的作者也在Issues
中回覆下個版本會增加單獨擴展名變數,所以文章的解決方案僅僅做一個分享,大家等Cloudreve
版本更新會更好。