精品欧美一区二区三区在线观看 _久久久久国色av免费观看性色_国产精品久久在线观看_亚洲第一综合网站_91精品又粗又猛又爽_小泽玛利亚一区二区免费_91亚洲精品国偷拍自产在线观看 _久久精品视频在线播放_美女精品久久久_欧美日韩国产成人在线

遠程熱部署的落地與思考-動態編譯篇

開發
遠程熱部署是參考美團Sonic并結合轉轉的業務場景研發的一款熱部署組件,由Java Agent與IDEA插件組成。整個熱部署全流程涉及知識范圍廣泛,三言兩語無法描述清楚,全流程會拆分成專題的形式進行分享。本文主要選講在落地過程中遇到的一些Sonic未提及的問題與自己的思考感悟。

遠程熱部署(代號名稱Mark42、Jarvis)是參考美團Sonic并結合轉轉的業務場景研發的一款熱部署組件,由Java Agent與IDEA插件組成。

整個熱部署全流程涉及知識范圍廣泛,三言兩語無法描述清楚,全流程會拆分成專題的形式進行分享。本文主要選講在落地過程中遇到的一些Sonic未提及的問題與自己的思考感悟。

通讀前建議閱讀美團原文:遠程熱部署在美團的落地實踐,原文講述到相關技術介紹、原理、實現方案等不再贅述。

1、背景

1.1 、真實工作場景

某次前后端聯調時的對話(部分內容存在虛構):

H師傅:果子果子,你這接口返回的結果好像不太對啊,是不是寫反了啊~

:啊,不能吧,稍等我看看哈~

H師傅:你看一下~

:woc,大于等于寫反了,我改一下~

H師傅:好小子,抓緊抓緊~

:改完了,就一個符號,已經在編譯部署了~

??五分鐘后......

:部署完了,你再看一下~

H師傅:好了,沒問題了~

H師傅:果子,這里返回的文案要不要把最后一句刪掉,不太通順~

:有道理,PM同意了,我刪一下~

??又過了五分鐘......

:編譯部署完了,你在看一下~

H師傅:可以的 可以的~


原本一兩分鐘可以完成的工作,由于代碼的改動、編譯部署等待導致前后端同學各自浪費了十多分鐘,極大的影響了協作效率。

如果能擁有一種“魔法”,使得后端的代碼像前端一樣“熱更新”,那該是一件多么幸福的事情!

1.2、項目背景

作為一名業務側的一線開發同學,一直把高優支持業務放在首位。由于業務系統相對復雜,且受限于公司架構歷史原因,使得開發者在開發過程中往往都是“一次性編寫”代碼,等業務邏輯實現的差不多,“看”上去沒問題,就部署到Docker容器中進行自測查漏補缺,當遇到極為復雜的場景,就需要進行遠程Debug協助,發現問題后修改代碼,再次部署,反反復復。

正因如此我們每天少不了Beetle(公司內部編譯管理及發布管理輕量級效率平臺)多次編譯與部署的循環反復的操作,一行小小的代碼改動就需要走完一整個流程才能使得代碼生效,嚴重影響了開發自測、聯調、提測的效率。

圖片

現有流程

面對如此“長”的流程,能否對其進行簡化,盡可能的減少編譯部署次數,使得修改后的代碼快速生效,減少用戶等待時間。

圖片

期望流程

2、預期目標

日常開發場景中,最大限度的幫助開發者減少代碼提交、編譯、部署的次數,節省因等待而造成的碎片化時間,使得開發者只需把主要精力放在編碼實現,間接提升開發效率。

3、選講問題分析

“熱部署”簡單講就是在Java程序運行時更新Java類文件,即JVM的字節碼重載,通過新的字節碼二進制流和舊的Class對象生成ClassDefinition定義,同時重載或初始化Spring容器以及第三方框架,達到“不停機”狀態更新

思考一個問題:新的字節碼二進制流也就是字節碼文件(.class 文件)從何而來呢?

無非存在兩種解法:

1、本地編譯Java源代碼,將生成的.class文件推到遠端服務器;

2、直接將Java源代碼推到遠端服務器,由遠端服務器進行編譯生成.class文件;

我們來逐一解析兩種方案成本與利弊:

方案1:成本低,易實現,用戶在本地先執行編譯操作,通過IDEA開發工具完成,但由于IDEA工具和Maven等構建工具之間的兼容性問題,經常出現本地編譯不通過的情況,當然也可以通過Maven的Install命令編譯整個服務文件,但是這種方案操作時間長,不人性化。其次還存在潛在的安全性問題:本地開發Jdk環境與服務器Jdk環境不一致等。

圖片

本地編譯失敗

方案2:難度系數高,實現復雜,但卻是更優解。首先由用戶將修改后的Java源代碼推到遠端服務器,由遠端服務器進行動態編譯生成.class文件,整個過程對用戶透明。

問題:

①極多數服務都是Springboot - Fat Jar(將一個Jar及其依賴的三方Jar全部打到一個包中,這個包即為FatJar)這種結構方式。想要動態編譯則需要從ClassLoader中恢復ClassPath,但Springboot - Fat Jar是一個整體的jar包,恢復出來的路徑不合法(Url轉換成File不存在),這就導致動態編譯時找不到代碼中引用的各種類。

圖片

Fat-Jar

②LomBok依賴丟失問題:Lombok主要是在編譯.class文件期間,生成Get/Set/Hash/Equals/ToString等方法,使實體對象更簡潔,所以像Lombok這樣的依賴只作用于編譯階段,編譯完成就沒用了,對于有“代碼潔癖”的同學會選擇從依賴Jar包里排除掉。這樣子可能會導致我們修改、新增實體類時動態編譯失敗,找不到依賴。

Maven如下配置:

<dependency>
    <groupId>org.projectlombok</groupId>
    <artifactId>lombok</artifactId>
    <version>1.18.6</version>
    <scope>provided</scope>
</dependency>
 
 
<plugins>
     <plugin>
         <groupId>org.springframework.boot</groupId>
         <artifactId>spring-boot-maven-plugin</artifactId>
         <configuration>
             <excludes>
                 <exclude>
                     <groupId>org.projectlombok</groupId>
                     <artifactId>lombok</artifactId>
                 </exclude>
             </excludes>
         </configuration>
     </plugin>
 </plugins>

③動態編譯時ClassLoader的處理。美團熱部署作者龍哥說:所有的遠程和本地執行不一致的問題,百分之99在ClassLoader的問題上找。動態編譯需兼容目前公司現有服務類型以及后續可能存在類型。

公司內部Java服務類型分為三種:SpringBoot服務、SCF服務、ZZJava服務,不同服務類型打包方式不同。

SpringBoot:Spring Boot服務,LaunchedURLClassLoader加載依賴資源

SCF服務:歷史SCF(內部RPC)框架內嵌Spring服務模式,服務啟動前需解壓服務所有依賴,得到絕對路徑后作為-classpath參數,通過AppClassLoader加載依賴資源(查看完整啟動命令足有2-3W字符,可怕??)

ZZJava服務:基于SpringBoot自定義的一種項目結構、打包及啟動、停止標準,依舊為Spring Boot服務,通過LaunchedURLClassLoader加載依賴資源

4、方案選擇

方案1:每次打包Docker鏡像時添加Dockerfile命令,解壓服務Jar包到指定位置,獲得BOOT-INF絕對路徑,并在JVM啟動命令中添加絕對路徑參數,服務運行時可取得BOOT-INF絕對路徑,并將其作為options -classpath參數調用getTask方法編譯代碼。

CompilationTask getTask(Writer out,
                            JavaFileManager fileManager,
                            DiagnosticListener<? super JavaFileObject> diagnosticListener,
                            Iterable<String> options,
                            Iterable<String> classes,
                            Iterable<? extends JavaFileObject> compilationUnits);

實現方案雖簡單,但目前架構不支持自定義Dockerfile命令,無法做到通用解壓服務,且依然無法解決像Lombok、Mapstruct等問題,某些情景下編譯還會報錯,可用性低。

方案2:解決Fatjar模式下的動態編譯

思考一下SpringBoot服務為什么可以讀取Fatjar的資源

一句話描述可以總結為SpringBoot自定義了URL Handler處理邏輯,將嵌套的jar轉換為URL,通過URLClassLoader的addURL方法添加獲取資源,完整細節可以翻閱SpringBoot源碼查看。

public URL getUrl() throws MalformedURLException {
        if (this.url == null) {
            String file = this.rootFile.getFile().toURI() + this.pathFromRoot + "!/";
            file = file.replace("file:////", "file://"); // Fix UNC paths
            // 這里返回的時候 new了一個Handler來處理URL
            this.url = new URL("jar", "", -1, file, new Handler(this));
        }
        return this.url;
    }

Handler繼承了URLStreamHandler,重寫了openConnection方法來處理獲取JarURLConnection,最終通過JarURLConnection的getInputStream方法返回字節流。

@Override
    protected URLConnection openConnection(URL url) throws IOException {
        if (this.jarFile != null && isUrlInJarFile(url, this.jarFile)) {
            return JarURLConnection.get(url, this.jarFile);
        }
        try {
            return JarURLConnection.get(url, getRootJarFileFromUrl(url));
        } catch (Exception ex) {
            return openFallbackConnection(url, ex);
        }
    }

我們回到URLClassLoader,URLClassLoader重寫了findClass方法,通過雙親委托加載資源

protected Class<?> findClass(final String name) throws ClassNotFoundException
    {
        final Class<?> result;
        try {
            result = AccessController.doPrivileged(
                new PrivilegedExceptionAction<Class<?>>() {
                    public Class<?> run() throws ClassNotFoundException {
                        String path = name.replace('.', '/').concat(".class");
                        // 這里調用URLClassPath的getResource方法
                        Resource res = ucp.getResource(path, false);
                        if (res != null) {
                            try {
                                return defineClass(name, res);
                            } catch (IOException e) {
                                throw new ClassNotFoundException(name, e);
                            } catch (ClassFormatError e2) {
                                if (res.getDataError() != null) {
                                    e2.addSuppressed(res.getDataError());
                                }
                                throw e2;
                            }
                        } else {
                            return null;
                        }
                    }
                }, acc);
        } catch (java.security.PrivilegedActionException pae) {
            throw (ClassNotFoundException) pae.getException();
        }
        if (result == null) {
            throw new ClassNotFoundException(name);
        }
        return result;
    }

最終調用到URLClassPath的getResource方法

Resource getResource(final String name, boolean check) {
            final URL url;
            try {
                url = new URL(base, ParseUtil.encodePath(name, false));
            } catch (MalformedURLException e) {
                throw new IllegalArgumentException("name");
            }
            final URLConnection uc;
            try {
                if (check) {
                    URLClassPath.check(url);
                }
                // 這里就會調用到URLStreamHandler的openConnection方法
                uc = url.openConnection();
                InputStream in = uc.getInputStream();
                if (uc instanceof JarURLConnection) {
                    /* Need to remember the jar file so it can be closed
                     * in a hurry.
                     */
                    JarURLConnection juc = (JarURLConnection)uc;

                    boolean firstLoad = jarfile == null;

                    jarfile = JarLoader.checkJar(juc.getJarFile());

                    if (firstLoad && JarLoadEvent.isEnabled()) {
                        Tooling.notifyEvent(JarLoadEvent.jarLoadEvent(url, jarfile));
                    }
                }
            } catch (Exception e) {
                return null;
            }
            return new Resource() {
                public String getName() { return name; }
                public URL getURL() { return url; }
                public URL getCodeSourceURL() { return base; }
                public InputStream getInputStream() throws IOException {
                    //JarURLConnection的getInputStream方法
                    return uc.getInputStream();
                }
                public int getContentLength() throws IOException {
                    return uc.getContentLength();
                }
            };
        }

既然SpringBoot已經幫我們處理好Fatjar的資源讀取,我們將直接復用其能力獲取加載的資源。

5、探索實踐

Agent啟動時,通過字節碼增強Spring框架。在Spring框架初始化時獲取其ClassLoader并反射存儲到Agent全局靜態字段(SpringBoot服務為LaunchedURLClassLoader,SCF服務為AppClassLoader)。當觸發動態編譯時(Agent運行期),針對于SpringBoot服務,我們將復用SpringBoot解析Fatjar的這個能力,通過LaunchedURLClassLoader獲取完整的URL資源,通過URL解析來得到JavaFileObject,從而完成動態編譯。

針對于缺失的Lombok、Mapstruct等依賴以及自定義添加的jar包,我們可以手動添加URL資源。

public DynamicCompiler(ClassLoader userClassLoader) {
        if (javaCompiler == null) {
            throw new IllegalStateException("Can not load JavaCompiler from javax.tools.ToolProvider#getSystemJavaCompiler(), please confirm the application running in JDK not JRE.");
        }

        standardFileManager = javaCompiler.getStandardFileManager(null, null, null);

        options.add("-Xlint:unchecked");
        options.add("-g");

        List<URL> urlList = new ArrayList<>();
        //添加自定義jar資源
        urlList.addAll(getCustomJarUrl());
        //獲取userClassLoader加載的資源(SpringBoot服務 LaunchedURLClassLoader)
        urlList.addAll(getClassLoaderUrl(userClassLoader));

        // 向上查找父類
        ClassLoader appClassLoader = getAppClassLoader(userClassLoader);

        //DynamicClassLoader同樣繼承URLClassLoader
        dynamicClassLoader = new DynamicClassLoader(urlList.toArray(new URL[0]), appClassLoader);
    }

解析URL獲取JavaFileObject

private List<JavaFileObject> processJar(URL packageFolderURL) {
        List<JavaFileObject> result = new ArrayList<>();
        try {
            String jarUri = packageFolderURL.toExternalForm().substring(0, packageFolderURL.toExternalForm().lastIndexOf("!/"));

            JarURLConnection jarConn = (JarURLConnection) packageFolderURL.openConnection();
            String rootEntryName = jarConn.getEntryName();
            if (StringUtils.isBlank(rootEntryName)){
                return new ArrayList<>();
            }

            int rootEnd = rootEntryName.length() + 1;

            Enumeration<JarEntry> entryEnum = jarConn.getJarFile().entries();
            while (entryEnum.hasMoreElements()) {
                JarEntry jarEntry = entryEnum.nextElement();
                String name = jarEntry.getName();
                if (name.startsWith(rootEntryName) && name.indexOf('/', rootEnd) == -1 && name.endsWith(CLASS_FILE_EXTENSION)) {
                    URI uri = URI.create(jarUri + "!/" + name);
                    String binaryName = name.replaceAll("/", ".");
                    binaryName = binaryName.replaceAll(CLASS_FILE_EXTENSION + "$", "");

                    result.add(new CustomJavaFileObject(binaryName, uri));
                }
            }
        } catch (Exception e) {
            throw new RuntimeException("Wasn't able to open " + packageFolderURL + " as a jar file", e);
        }
        return result;
    }

動態編譯獲取字節碼

public Map<String, byte[]> buildGetByteCodes() {

        errors.clear();
        warnings.clear();

        JavaFileManager fileManager = new DynamicJavaFileManager(standardFileManager, dynamicClassLoader);

        DiagnosticCollector<JavaFileObject> collector = new DiagnosticCollector<>();
        JavaCompiler.CompilationTask task = javaCompiler.getTask(null, fileManager, collector, options, null, compilationUnits);

        try {

            if (!compilationUnits.isEmpty()) {
                boolean result = task.call();

                if (!result || collector.getDiagnostics().size() > 0) {

                    for (Diagnostic<? extends JavaFileObject> diagnostic : collector.getDiagnostics()) {
                        switch (diagnostic.getKind()) {
                            case NOTE:
                            case MANDATORY_WARNING:
                            case WARNING:
                                warnings.add(diagnostic);
                                break;
                            case OTHER:
                            case ERROR:
                            default:
                                errors.add(diagnostic);
                                break;
                        }

                    }
                    if (!errors.isEmpty()) {
                        return new HashMap<>();
                    }
                }
            }

            return dynamicClassLoader.getByteCodes();
        } catch (ClassFormatError e) {
            throw new DynamicCompilerException(e, errors);
        } finally {
            compilationUnits.clear();
        }
    }

Mapstruct編譯過程較為特殊,首先會根據接口生成接口的實現類,進而生成字節碼,getJavaFileForOutput方法需要根據kind類型判斷一下,不能忽略SOURCE類型,不然會導致Mapstruct接口的字節碼文件里存儲的是實現類的Java代碼,進而導致JVM的字節碼重載錯誤。

@Override
    public JavaFileObject getJavaFileForOutput(Location location, String className, JavaFileObject.Kind kind, FileObject sibling) {

        if (JavaFileObject.Kind.SOURCE.equals(kind)) {
            // 源碼
            for (StringSource stringSource : this.sourceCodes) {
                if (stringSource.getClassName().equals(className)) {
                    return stringSource;
                }
            }

            StringSource stringSource = new StringSource(className);
            sourceCodes.add(stringSource);
            //這里可以存一下動態生成的源代碼,編譯完成后輸出到文件夾
            //            classLoader.registerCompiledSource(stringSource);
            return stringSource;

        } else {
            // 字節碼
            for (MemoryByteCode byteCode : this.byteCodes) {
                if (byteCode.getClassName().equals(className)) {
                    return byteCode;
                }
            }

            MemoryByteCode innerClass = new MemoryByteCode(className);
            byteCodes.add(innerClass);
            classLoader.registerCompiledSource(innerClass);
            return innerClass;
        }
    }

注:動態編譯時一定要添加-g參數生成完整調試信息,不然熱部署代碼Debug會發現方法棧內變量沒有名字、Jacoco布爾數組透出、slot對不上等問題。(坑了我半年多一直沒發現原因)

做完以上動作,你就可以任意的動態編譯Java源代碼,得到字節碼文件了。

到這就完成了遠程熱部署準備工作了。

6、總結與展望

經常被問到做熱部署的夙愿是什么:

遠程熱部署的初心不是代替掉Beetle發布部署流程,而是盡可能減少用戶編譯部署次數,節省用戶碎片化的時間,希望可以做到一次部署,“任意”修改

初版交互圖:

圖片

部分功能UI交互展示圖:圖片


目前Mark42已經支持以下功能

框架/功能

狀態

遠程熱部署

?

遠程動態編譯

?

熱部署代碼遠程Debug

?

遠程Agent日志

?

遠程服務日志

?

批量熱部署

?

IDEA插件集成

?

修改方法體內容

?

新增方法體

?

新增泛型方法

?

新增非靜態字段

?

新增修改靜態字段

?

新增修改繼承類

?

新增修改接口方法

?

新增修改匿名內部類

?

新增修改靜態塊

?

FastJson

?

Jackson

?

Jdk代理

?

Spring

?

Spring MVC

?

Avenger

?

Fsmx狀態機

?

ZZMQ

?

MyBatis

?

Mapstruct

?

XXL-JOB

?

SCF

?

......



熱部署還有很長的路要走,跟美團Sonic相比,這僅僅是剛開始

ToDoList:

框架/功能

狀態

Configuration配置bean支持

已支持、測試中

xml文件配置bean支持

已支持、測試中

SCF Agent級別調用

待開發

遠程單測支持

待開發

遠程反編譯

待開發

究極體 Spring loader替換dcevm

待開發

關于作者

譚金果 22屆校招生,現任轉轉B2C技術部供應鏈后端研發工程師

責任編輯:龐桂玉 來源: 轉轉技術
相關推薦

2022-03-17 21:42:20

美團插件技術

2016-10-21 15:58:51

容器容器技術Docker

2020-03-09 15:40:27

RSACDevSecOps懸鏡安全

2016-11-07 18:27:32

云計算阿里云

2024-06-17 08:22:31

GenAI技術人工智能

2011-12-05 12:59:22

JavaJVM

2022-04-13 12:20:21

零信任網絡安全

2018-07-19 06:23:37

物聯網IOT網絡

2022-02-09 15:02:01

ideadocker遠程熱部署

2020-08-03 11:38:08

數據中臺數據平臺互聯網

2018-06-27 15:59:33

2016-12-29 14:06:08

SDNFV技術商用部署

2021-11-08 10:24:05

智能自動駕駛汽車

2020-07-03 19:48:24

區塊鏈區塊鏈技術

2019-11-08 08:53:26

HDFS監控架構

2018-01-18 06:34:46

2022-01-21 16:14:41

汽車底盤技術

2019-08-26 11:03:32

互聯網研發落地

2021-10-28 10:10:44

人工智能

2010-06-18 15:03:12

BGP路由協議
點贊
收藏

51CTO技術棧公眾號

国产又黄又粗又猛又爽的| 欧美黄网站在线观看| 国产不卡精品视频| 在线日韩欧美| 亚洲人线精品午夜| 亚洲第一区第二区第三区| av福利导福航大全在线| 久久久国际精品| 亚洲最大的av网站| 国产精品xxxx喷水欧美| 精品72久久久久中文字幕| 欧美群妇大交群的观看方式| 日韩 欧美 视频| av在线播放网| 国产精品一区二区免费不卡 | 国产原创欧美精品| 在线看成人av| 国产国产精品| 亚洲男人av在线| 久久精品无码一区二区三区毛片| 毛片电影在线| 亚洲综合久久久久| 亚洲精品tv久久久久久久久| 日本韩国在线观看| 久久国产精品色婷婷| 欧美一级大片在线免费观看| 成人自拍小视频| 国产日韩欧美一区二区三区| 精品久久人人做人人爽| 爱爱爱爱免费视频| 日韩a**中文字幕| 亚洲妇女屁股眼交7| 中文字幕一区二区三区四区五区六区| 日韩精品视频在线观看一区二区三区| 国产一区三区三区| 国产精品普通话| 69国产精品视频免费观看| 狠狠噜噜久久| 欧美xxxx18国产| 丁香激情五月少妇| 亚洲另类av| 日韩av中文字幕在线| 国产精品熟妇一区二区三区四区| 一区二区三区日本视频| 在线观看一区日韩| 国产精品免费成人| 不卡福利视频| 日韩欧美在线播放| 精品国产一区三区| 丁香花高清在线观看完整版| 亚洲国产日韩在线一区模特| 久久www视频| 日本资源在线| 亚洲一区二区五区| 久久99久久99精品| 岛国av在线网站| 午夜精品国产更新| 亚洲熟妇国产熟妇肥婆| av影院在线免费观看| 亚洲国产成人高清精品| 丰满的少妇愉情hd高清果冻传媒 | 亚洲国产精品欧美久久| 国产精品一区二区91| 91免费的视频在线播放| 一本一道人人妻人人妻αv| 蜜臀av在线播放一区二区三区| 日本电影亚洲天堂| 黄色av一区二区| 美女尤物国产一区| 成人av资源在线播放| 国产男男gay体育生网站| 国产精品亚洲第一区在线暖暖韩国| 亚洲在线免费视频| 国产综合在线播放| 91免费精品国自产拍在线不卡 | 郴州新闻综合频道在线直播| 在线视频欧美日韩精品| 国产麻豆a毛片| 欧美黄色一区二区| 午夜免费久久久久| 国产又大又黄又粗| 精品综合久久久久久8888| 亚洲va电影大全| 亚洲高清视频网站| 26uuu欧美| 亚洲一区二区高清视频| 91三级在线| 精品免费在线观看| 美女网站色免费| 色悠久久久久综合先锋影音下载| 亚洲国产精品中文| 少妇精品无码一区二区免费视频| 中文字幕日韩欧美精品高清在线| 久久免费高清视频| 亚洲视屏在线观看| 国产成人av电影在线| 欧美三日本三级少妇三99| 秋霞午夜在线观看| 香蕉成人啪国产精品视频综合网 | 国产午夜视频在线| 日日摸夜夜添夜夜添国产精品| 成人免费午夜电影| 日韩有码第一页| 国产精品毛片a∨一区二区三区 | 老司机免费视频久久| 91视频国产一区| 日韩资源在线| 依依成人精品视频| 十八禁视频网站在线观看| 国产精品99久久免费| 精品视频在线播放色网色视频| 成年人网站在线观看视频| 99在线|亚洲一区二区| 国产男人精品视频| 天堂av电影在线观看| 国产精品二三区| 国产特级黄色大片| 亚洲视频国产| www.国产一区| 久草视频一区二区| 成人动漫一区二区三区| 宅男av一区二区三区| 欧美美女日韩| 精品国产a毛片| 国产精品国产精品88| 日本成人超碰在线观看| 久久久久免费网| 手机在线免费av| 欧美久久久久久久久久| 欧美三级视频网站| 老司机午夜精品视频在线观看| 狠狠色狠狠色综合人人| 色女人在线视频| 91精品久久久久久久91蜜桃 | 亚洲精品国产精品国| ㊣最新国产の精品bt伙计久久| 国产成人av影视| 色婷婷狠狠五月综合天色拍 | 黄页视频在线91| 日韩一二三区不卡在线视频| 中日韩脚交footjobhd| 亚洲成人国产精品| 久久久综合久久| 福利一区二区在线观看| 中文字幕在线乱| 亚洲国产伊人| 久久久国产精彩视频美女艺术照福利| 中国女人一级一次看片| 国产欧美综合在线观看第十页| 欧美日韩在线视频一区二区三区| 青青久久av| 欧洲成人性视频| 三级在线视频| 一本久久a久久免费精品不卡| 波多野结衣av在线免费观看| 校园激情久久| 日本在线观看一区二区| 成人开心激情| 最近2019免费中文字幕视频三 | 精品三级在线观看| 国产午夜精品无码| xfplay精品久久| 老熟妇仑乱视频一区二区| 精品国产乱码久久久| 国产精品扒开腿爽爽爽视频| aⅴ在线视频男人的天堂| 欧美日韩国产片| 动漫性做爰视频| 成人精品国产福利| 丰满少妇被猛烈进入高清播放| 亚洲香蕉视频| 国产精品中文在线| 性xxxfreexxxx性欧美| 亚洲黄色av女优在线观看| 9i看片成人免费看片| 欧美国产综合一区二区| 日韩欧美中文视频| 亚洲国产免费看| 日韩av电影免费在线观看| **国产精品| 国内精品久久久久伊人av | 久久亚洲精品小早川怜子| 成人性生生活性生交12| 91成人免费| 国语精品免费视频| 日韩在线短视频| 久久综合久久美利坚合众国| 日韩永久免费视频| 欧美男男青年gay1069videost| 久草免费在线视频观看| 久久综合狠狠综合久久激情| 国模私拍视频在线观看| 亚洲视频久久| 午夜精品亚洲一区二区三区嫩草 | 日韩一卡二卡三卡四卡| 国产精品美女久久久久av爽| 国产精品成人网| 最近日本中文字幕| 国内精品伊人久久久久av一坑 | 日本sm残虐另类| 性刺激综合网| 国产精品18hdxxxⅹ在线| 国产成人av在线播放| 少妇视频在线| 中文字幕日韩欧美精品在线观看| 国精产品乱码一区一区三区四区| 欧美系列在线观看| 五月天婷婷网站| 自拍偷拍亚洲欧美日韩| 国产精品美女高潮无套| 成人午夜在线视频| 国产乱女淫av麻豆国产| 美女尤物久久精品| 欧美国产视频一区| 成人中文在线| 精品无人乱码一区二区三区的优势 | 亚洲精品久久久久avwww潮水| 欧美三级在线播放| 你懂的国产在线| 亚洲国产精品影院| 欧美日韩中文字幕在线观看| 欧美极品aⅴ影院| aa片在线观看视频在线播放| 国产成人免费xxxxxxxx| 亚洲欧美自偷自拍另类| 美女日韩在线中文字幕| 91成人在线观看喷潮教学| 欧美.www| 大地资源第二页在线观看高清版| 精品国产一区二区三区av片| 国产在线一区二区三区四区| **爰片久久毛片| 91香蕉嫩草影院入口| 黄色精品视频网站| 国产精品99久久久久久白浆小说 | 视频一区二区三区国产| 亚洲国产美女久久久久| 亚洲黄色在线观看视频| 日韩欧美在线观看一区二区三区| 国产尤物在线观看| 欧美日韩中文一区| 中文字幕 自拍偷拍| 色香色香欲天天天影视综合网| 日产电影一区二区三区| 亚洲图片欧美色图| 黄色小视频在线免费看| 亚洲电影在线播放| 日韩福利片在线观看| 亚洲自拍偷拍综合| 日韩免费一二三区| 香蕉成人伊视频在线观看| 日韩三级小视频| 精品久久久久久久久久久| 日韩欧美大片在线观看| 五月天国产精品| 久久精品视频1| 色综合久久综合中文综合网| 无码人妻av一区二区三区波多野| 日本精品免费观看高清观看| 中文字幕永久免费视频| 在线91免费看| 国产suv一区二区| 精品国产乱子伦一区| 午夜在线观看视频18| 亚洲跨种族黑人xxx| 国产一级在线| 久久精品国产精品| 色av手机在线| 日本精品视频在线播放| 99re久久| 成人字幕网zmw| 精品久久ai电影| 久热这里只精品99re8久| 青草国产精品| 亚洲激情免费视频| 亚欧成人精品| 一区二区在线免费看| 国产91高潮流白浆在线麻豆| 日本少妇色视频| 国产精品国产成人国产三级| 欧美精品xxxxx| 色域天天综合网| 国产乱人乱偷精品视频a人人澡| 精品国产麻豆免费人成网站| 国产在线黄色| 欧美疯狂性受xxxxx另类| 新版的欧美在线视频| 国产日韩av在线播放| 国产成人夜色高潮福利影视| 日韩av大全| 狠狠色狠狠色综合日日tαg| 能看的毛片网站| 国产91在线|亚洲| 少妇av片在线观看| 午夜久久久久久| 91在线公开视频| 亚洲第一精品夜夜躁人人爽| a中文在线播放| 欧美激情2020午夜免费观看| 日韩免费电影| 成人资源av| 97精品97| 成人久久久久久久久| 国产精品一区二区在线观看网站| 免费看污片的网站| 亚洲国产精品久久久男人的天堂| 日批视频免费观看| 亚洲国产欧美自拍| 久cao在线| 国产不卡av在线免费观看| 亚洲1区在线| 一级特黄录像免费播放全99| 亚洲人成在线影院| 九九九久久久久久久| 久久精品视频免费| 动漫精品一区一码二码三码四码| 欧美猛男男办公室激情| 男女污视频在线观看| 欧美激情手机在线视频 | 91精品国产91久久久久久最新毛片| 香港一级纯黄大片| 中文字幕欧美日韩精品| а√天堂中文资源在线bt| 97人人澡人人爽| 91精品国产乱码久久久久久久| 青青青在线视频免费观看| 91一区在线观看| 豆国产97在线 | 亚洲| 日韩欧美一二区| 国产精品刘玥久久一区| 国产精品99久久久久久人| 色狼人综合干| 欧洲黄色一级视频| 成人综合婷婷国产精品久久免费| 91插插插插插插| 欧美酷刑日本凌虐凌虐| 免费黄网站在线播放| 国产精品情侣自拍| 日韩欧美视频专区| 五月天亚洲视频| 国产女主播视频一区二区| 91在线视频在线观看| 亚洲另类激情图| 高清不卡av| 久久免费视频1| 男人天堂欧美日韩| 扒开jk护士狂揉免费| 色偷偷成人一区二区三区91| 日本一区二区三区在线观看视频| 欧美亚洲国产视频| 夜夜春成人影院| 青青草精品视频在线观看| 中文字幕精品一区二区三区精品| www.久久网| 中文字幕亚洲欧美| 亚洲精品aa| 黄色一级大片免费| av欧美精品.com| 亚洲GV成人无码久久精品| 一个人www欧美| 96sao精品免费视频观看| 黄黄视频在线观看| 成人丝袜视频网| 国产三级精品三级在线观看| 一区二区国产精品视频| 四虎精品一区二区免费| 91视频 - 88av| 99热精品一区二区| 久久久久久不卡| 日韩中文在线不卡| 日韩视频一二区| 日本a在线免费观看| 久久久亚洲高清| 中文字幕第2页| 欧美理论电影在线观看| 国产在线播放精品| 无码日韩人妻精品久久蜜桃| 自拍偷拍欧美精品| 秋霞网一区二区| 国产精品久久婷婷六月丁香| 一区二区三区四区日韩| 精品少妇人妻av一区二区三区| 日本国产一区二区| av免费在线观看网址| 国产伦精品一区二区三区免| 日韩精品国产精品| 黄色一级片中国| 亚洲精品视频免费在线观看| 欧美性www| 日韩精品 欧美| 国产精品丝袜久久久久久app| 国内精品国产成人国产三级| 欧洲亚洲女同hd| 欧美在线亚洲综合一区| 麻豆国产精品一区| 91精品啪在线观看国产60岁| 波多野结衣久久精品| 欧美在线观看黄| 中文字幕欧美日韩一区| 天堂8在线视频|