Spring Boot打包JAR或WAR后,Mybatis的Mapper无法扫描

https://github.com/mybatis/mybatis-3/issues/325

https://github.com/StripesFramework/stripes/issues/35

MyBatis扫描通过VFS来实现

在Spring Boot中,由于是嵌套Jar,导致Mybatis默认的VFS实现DefaultVFS无法扫描嵌套Jar中的类。

解决办法,实现自定义的VFS,参考DefaultVFS增加对Spring Boot嵌套JAR的处理。

package com.qiyun.mybatis;

import java.io.BufferedReader;

import java.io.File;

import java.io.FileNotFoundException;

import java.io.IOException;

import java.io.InputStream;

import java.io.InputStreamReader;

import java.io.UnsupportedEncodingException;

import java.net.MalformedURLException;

import java.net.URL;

import java.net.URLEncoder;

import java.util.ArrayList;

import java.util.Arrays;

import java.util.List;

import java.util.jar.JarEntry;

import java.util.jar.JarInputStream;

import org.apache.ibatis.io.VFS;

import org.apache.ibatis.logging.Log;

import org.apache.ibatis.logging.LogFactory;

/**

* Modified DefaultVFS for handling nested jar.

*

* https://github.com/mybatis/mybatis-3/issues/325

* https://github.com/StripesFramework/stripes/issues/35

*

* @version 1.0

* @since 1.0

*/

public class SpringBootExecutableJarVFS extends VFS {

private static final Log log = LogFactory.getLog(SpringBootExecutableJarVFS.class);

/** The magic header that indicates a JAR (ZIP) file. */

private static final byte[] JAR_MAGIC = { 'P', 'K', 3, 4 };

@Override

public boolean isValid() {

return true;

}

@Override

public List list(URL url, String path) throws IOException {

InputStream is = null;

try {

List resources = new ArrayList();

// First, try to find the URL of a JAR file containing the requested resource. If a JAR

// file is found, then we'll list child resources by reading the JAR.

URL jarUrl = findJarForResource(url);

if (jarUrl != null) {

is = jarUrl.openStream();

if (log.isDebugEnabled()) {

log.debug("Listing " + url);

}

resources = listResources(new JarInputStream(is), path);

} else {

List children = new ArrayList();

try {

if (isJar(url)) {

// Some versions of JBoss VFS might give a JAR stream even if the resource

// referenced by the URL isn't actually a JAR

is = url.openStream();

JarInputStream jarInput = new JarInputStream(is);

if (log.isDebugEnabled()) {

log.debug("Listing " + url);

}

for (JarEntry entry; (entry = jarInput.getNextJarEntry()) != null;) {

if (log.isDebugEnabled()) {

log.debug("Jar entry: " + entry.getName());

}

children.add(entry.getName());

}

jarInput.close();

} else {

/*

* Some servlet containers allow reading from directory resources like a

* text file, listing the child resources one per line. However, there is no

* way to differentiate between directory and file resources just by reading

* them. To work around that, as each line is read, try to look it up via

* the class loader as a child of the current resource. If any line fails

* then we assume the current resource is not a directory.

*/

is = url.openStream();

BufferedReader reader = new BufferedReader(new InputStreamReader(is));

List lines = new ArrayList();

for (String line; (line = reader.readLine()) != null;) {

if (log.isDebugEnabled()) {

log.debug("Reader entry: " + line);

}

lines.add(line);

if (getResources(path + "/" + line).isEmpty()) {

lines.clear();

break;

}

}

if (!lines.isEmpty()) {

if (log.isDebugEnabled()) {

log.debug("Listing " + url);

}

children.addAll(lines);

}

}

} catch (FileNotFoundException e) {

/*

* For file URLs the openStream() call might fail, depending on the servlet

* container, because directories can't be opened for reading. If that happens,

* then list the directory directly instead.

*/

if ("file".equals(url.getProtocol())) {

File file = new File(url.getFile());

if (log.isDebugEnabled()) {

log.debug("Listing directory " + file.getAbsolutePath());

}

if (file.isDirectory()) {

if (log.isDebugEnabled()) {

log.debug("Listing " + url);

}

children = Arrays.asList(file.list());

}

} else {

// No idea where the exception came from so rethrow it

throw e;

}

}

// The URL prefix to use when recursively listing child resources

String prefix = url.toExternalForm();

if (!prefix.endsWith("/")) {

prefix = prefix + "/";

}

// Iterate over immediate children, adding files and recursing into directories

for (String child : children) {

String resourcePath = path + "/" + child;

resources.add(resourcePath);

URL childUrl = new URL(prefix + child);

resources.addAll(list(childUrl, resourcePath));

}

}

return resources;

} finally {

if (is != null) {

try {

is.close();

} catch (Exception e) {

// Ignore

}

}

}

}

/**

* List the names of the entries in the given {@link JarInputStream} that begin with the

* specified {@code path}. Entries will match with or without a leading slash.

*

* @param jar The JAR input stream

* @param path The leading path to match

* @return The names of all the matching entries

* @throws IOException If I/O errors occur

*/

protected List listResources(JarInputStream jar, String path) throws IOException {

// Include the leading and trailing slash when matching names

if (!path.startsWith("/")) {

path = "/" + path;

}

if (!path.endsWith("/")) {

path = path + "/";

}

// Iterate over the entries and collect those that begin with the requested path

List resources = new ArrayList();

for (JarEntry entry; (entry = jar.getNextJarEntry()) != null;) {

if (!entry.isDirectory()) {

// Add leading slash if it's missing

String name = entry.getName();

if (!name.startsWith("/")) {

name = "/" + name;

}

// Check file name

if (name.startsWith(path)) {

if (log.isDebugEnabled()) {

log.debug("Found resource: " + name);

}

// Trim leading slash

resources.add(name.substring(1));

}

}

}

return resources;

}

/**

* Attempts to deconstruct the given URL to find a JAR file containing the resource referenced

* by the URL. That is, assuming the URL references a JAR entry, this method will return a URL

* that references the JAR file containing the entry. If the JAR cannot be located, then this

* method returns null.

*

* @param url The URL of the JAR entry.

* @return The URL of the JAR file, if one is found. Null if not.

* @throws MalformedURLException

*/

protected URL findJarForResource(URL url) throws MalformedURLException {

if (log.isDebugEnabled()) {

log.debug("Find JAR URL: " + url);

}

if (isNestedJar(url)) {

// Retain jar: protocol as a workaround for #325

if (log.isDebugEnabled()) {

log.debug("It is a nested JAR: " + url);

}

} else {

// If the file part of the URL is itself a URL, then that URL probably points to the JAR

try {

for (;;) {

url = new URL(url.getFile());

if (log.isDebugEnabled()) {

log.debug("Inner URL: " + url);

}

}

} catch (MalformedURLException e) {

// This will happen at some point and serves as a break in the loop

}

}

// Look for the .jar extension and chop off everything after that

StringBuilder jarUrl = new StringBuilder(url.toExternalForm());

int index = jarUrl.lastIndexOf(".jar");

if (index >= 0) {

jarUrl.setLength(index + 4);

if (log.isDebugEnabled()) {

log.debug("Extracted JAR URL: " + jarUrl);

}

} else {

if (log.isDebugEnabled()) {

log.debug("Not a JAR: " + jarUrl);

}

return null;

}

// Try to open and test it

try {

URL testUrl = new URL(jarUrl.toString());

if (isJar(testUrl)) {

return testUrl;

} else {

// WebLogic fix: check if the URL's file exists in the filesystem.

if (log.isDebugEnabled()) {

log.debug("Not a JAR: " + jarUrl);

}

jarUrl.replace(0, jarUrl.length(), testUrl.getFile());

File file = new File(jarUrl.toString());

// File name might be URL-encoded

if (!file.exists()) {

try {

file = new File(URLEncoder.encode(jarUrl.toString(), "UTF-8"));

} catch (UnsupportedEncodingException e) {

throw new RuntimeException("Unsupported encoding?  UTF-8?  That's unpossible.");

}

}

if (file.exists()) {

if (log.isDebugEnabled()) {

log.debug("Trying real file: " + file.getAbsolutePath());

}

testUrl = file.toURI().toURL();

if (isJar(testUrl)) {

return testUrl;

}

}

}

} catch (MalformedURLException e) {

log.warn("Invalid JAR URL: " + jarUrl);

}

if (log.isDebugEnabled()) {

log.debug("Not a JAR: " + jarUrl);

}

return null;

}

protected boolean isNestedJar(URL url) {

if (!"jar".equals(url.getProtocol()))

return false;

String urlStr = url.toExternalForm();

int indexOfWar = urlStr.indexOf(".war!/");

int indexOfJar = urlStr.indexOf(".jar!/");

if (indexOfWar != -1) { // Executable War

return indexOfWar != urlStr.lastIndexOf(".jar!/");

} else { // Executable Jar

return indexOfJar != urlStr.lastIndexOf(".jar!/");

}

}

/**

* Converts a Java package name to a path that can be looked up with a call to

* {@link ClassLoader#getResources(String)}.

*

* @param packageName The Java package name to convert to a path

*/

protected String getPackagePath(String packageName) {

return packageName == null ? null : packageName.replace('.', '/');

}

/**

* Returns true if the resource located at the given URL is a JAR file.

*

* @param url The URL of the resource to test.

*/

protected boolean isJar(URL url) {

return isJar(url, new byte[JAR_MAGIC.length]);

}

/**

* Returns true if the resource located at the given URL is a JAR file.

*

* @param url The URL of the resource to test.

* @param buffer A buffer into which the first few bytes of the resource are read. The buffer

*            must be at least the size of {@link #JAR_MAGIC}. (The same buffer may be reused

*            for multiple calls as an optimization.)

*/

protected boolean isJar(URL url, byte[] buffer) {

InputStream is = null;

try {

is = url.openStream();

is.read(buffer, 0, JAR_MAGIC.length);

if (Arrays.equals(buffer, JAR_MAGIC)) {

if (log.isDebugEnabled()) {

log.debug("Found JAR: " + url);

}

return true;

}

} catch (Exception e) {

// Failure to read the stream means this is not a JAR

} finally {

if (is != null) {

try {

is.close();

} catch (Exception e) {

// Ignore

}

}

}

return false;

}

}

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,837评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,551评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,417评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,448评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,524评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,554评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,569评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,316评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,766评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,077评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,240评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,912评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,560评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,176评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,425评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,114评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,114评论 2 352

推荐阅读更多精彩内容