Jetty7 热部署过程详解

读者对象

    对 jetty 比较熟悉,想了解其部署原理的开发人员。


一、预备知识

1、什么是jetty
    官方描述:Jetty是一个100%由Java实现的、开源的HTTP服务器和javax.servlet容器,它不仅仅作为一个独立服务软件(如Tomcat)被使用,而且其优良的组件(Componet)设计、高内聚低耦合、高扩展性等特性使得Jetty非常易于作为嵌入式工具使用。由于Jetty构架优秀、实现优雅,所以它被广泛嵌入的到移动设备、工具、框架(frameworks)、应用程序服务器(Application Server)等等领域。

    从开发人员的视图看jetty,由三部分组成:
    A、start.jar:
一个很独立的jar,43k,包含7个类,主要作用是服务器启动前的准备工作。
    B、lib 下的一组jar
jetty的核心jar,每个jar提供不同的功能,如 jetty-jmx.jar 提供JMX功能,jetty-jndi 提供JNDI支持。
    C、etc 下的一组xml文件
Jetty提供了类似IoC/DI容器,jetty.xml配置文件就是这个容器的配置文件。

2、jetty 的过人之处
    简单、精巧、嵌入式
    最快的Servlet服务器之一
    可以处理上千个并发连接
    基于NIO的 Continuation
    websocket 支持

二、jetty 部署概述

1、部署方式介绍
    jetty 支持文件方式和目录方式。

2、部署过程描述

    先从jetty启动说起,通过 java -jar start.jar 启动之后,start.jar 主要做了如下工作:
    1、解析命令行参数,如 -version ,只是屏幕输出各个jar的版本信息,不会启动服务器
    2、准备所需的配置文件,默认配置文件在 start.ini 中指定,也可以在命令行中指定
    3、设置 classpath,根据 OPTIONS 指定的值,将对应的jar加入到classpath
    4、通过反射调用 start.class 指定的值,以启动服务器,start.class的值在 start.config 中指定,文件位于 start.jar\org\eclipse\jetty\start\start.config
默认的启动类是 org.eclipse.jetty.xml.XmlConfiguration,该类解析xml文件,解析的过程就是服务器启动的过程。
    org.eclipse.jetty.server.Server 实现了 LifeCycle,所以在 解析时,会调用 Server 的生命周期方法,在 Server 的 start 过程中,会设置线程池(ThreadPool)、连接器(Connector),并调用 dependentBeans 中所有 bean 的生命周期方法,其中有一个 bean 就是 DeploymentManager(部署管理器),在文件 jetty-deploy.xml 中定义。

    DeploymentManager 中添加了两个 AppProvider,分别是 ContextProvider 和 WebAppProvider。
    ContextProvider:用于部署/监控 contexts 目录,也可指定目录,通过 monitoredDir 属性指定,里面的文件必须是 xml ,并且 xml 中定义的类必须是 ContextHandler 的子类,一般用来定义一个 WebAppContext 实例,该实例对应一个应用(WAR)WebAppProvider:用于部署/监控 webapp 目录,也可指定目录,通过 monitoredDir 属性指定,里面的文件必须是.war或war目录。
    DeploymentManager 也 extends LifeCycle,在 start 过程中,会 1、添加部署过程监听器;2、调用注册的 AppProvider,部署应用。
    WebAppProvider 的部署:
    1、扫描 webapp 目录,满足条件的添加进 _currentScan(在Scanner中),其规则是:1、是一个标准的WAR目录;2、一个.war文件;3、在 contexts 目录中没有对应的xml文件。
    2、具体的扫描过程由 Scanner 完成,在方法 scan 中,完成扫描、部署动作,之后启动定时器,循环调用 scan ,以实现热更新。
    3、部署过程由 reportAddition 方法触发,在该方法中会触发 Scanner 类中定义的部署监听器,能监听三种状态:文件添加、删除、修改
    4、对于添加(即新增部署一个应用),调用 DeploymentManager 的 addApp 方法,在该方法中触发 requestAppGoal,即 Immediately attempt to go to default lifecycle state。
    5、应用的各种状态在jetty中用有向图来描述,其包括的状态有:starting,deployed,started,stopping,deploying,undeployed,undeploying 等。每种状态都定义的相应的绑定监听器,如 starting 过程定义了 StandardStarter,以负责具体的部署过程。
    6、最后调用 ServletContextHandler 的 startContext 开始部署应用:解析配置,完成 servlet 容器该完成的工作。
ContextProvider 的部署过程类似,它在解析xml过程中产生的 App,其中定义了war的位置,它会根据该war的位置去部署。


三、热部署过程详解

部署所涉及核心类的UML图:




    实际上,jetty热部署过程的实现相当的简陋,其大致思路如下:

    先对webapps目录进行扫描,确定需要部署的App,这里的App是contexts目录中没有的应用,即没有同名的xml文件,部署完这些App之后,用名为 _currentScan 的 Map存储,key是app的路径,value是该目录的最后修改时间,启动定时器,按照设置的间隔时间循环扫描。扫描时用新产生的 _currentScan 和 _prevScan(与_currentScan相同,表示前一次扫描的结果)进行比对,确定差异,添加、修改、删除,分别再触发不同的监听器。
    对于 添加 的应用,直接新增一个部署,对于 删除 的应用,直接删除一个部署,对于 修改 的应用,jetty的做法是先删除,再将应用完全重新部署。
    问题:只有当应用目录的最后修改时间改变了,jetty才认为是应用被修改了,再会重新部署应用,而修改应用中的文件是不会引起应用目录的修改时间改变的,所以,在一般情况下,webapps 中的应用不会被检查到修改。

    而对于contexts目录下的xml文件,和webapps目录一样,也是启动定时器定时扫描,找出更新过的文件,其策略是,如果文件的最后修改时间不同,则表示应该被更新了,则应重新部署应用。
    问题:任何一点小小的改动都能导致xml文件的修改时间变化,比如说在文件中加一个空格,就会导致应用被卸载,然后再被部署。难道不能做得更精致一点?
    对于以上的描述,有代码为证,简单罗列如下:

public class DeploymentManager extends AbstractLifeCycle
{
    ......
    protected void doStart() throws Exception
    {
        if (_useStandardBindings)//如果使用标准的应用生命周期绑定监听器,则添加jetty自带的,否则,用户可自义绑定监听器
        {
            Log.debug("DeploymentManager using standard bindings");
            addLifeCycleBinding(new StandardDeployer());
            addLifeCycleBinding(new StandardStarter());
            addLifeCycleBinding(new StandardStopper());
            addLifeCycleBinding(new StandardUndeployer());
        }

        // Start all of the AppProviders
        for (AppProvider provider : _providers)//默认 _providers 包括两个:ContextProvider 和 WebAppProvider
        {
            startAppProvider(provider);
        }
        super.doStart();
    }

    ......

    //触发应用整个生命周期绑定
    private void requestAppGoal(AppEntry appentry, String nodeName)
    {
        Node destinationNode = _lifecycle.getNodeByName(nodeName);
        // Compute lifecycle steps
        Path path = _lifecycle.getPath(appentry.lifecyleNode,destinationNode);
        if (path.isEmpty())
        {
            // nothing to do. already there.
            return;
        }

        // Execute each Node binding.  Stopping at any thrown exception.
        try
        {
            Iterator<Node> it = path.getNodes().iterator();
            if (it.hasNext()) // Any entries?
            {
                // The first entry in the path is always the start node
                // We don't want to run bindings on that entry (again)
                it.next(); // skip first entry
                while (it.hasNext())
                {
                    Node node = it.next();
                    Log.debug("Executing Node: " + node);
                    _lifecycle.runBindings(node,appentry.app,this);//运行绑定
                    appentry.setLifeCycleNode(node);
                }
            }
        }
        catch (Throwable t)
        {
            Log.warn("Unable to reach node goal: " + nodeName,t);
        }
    }

}

public abstract class ScanningAppProvider extends AbstractLifeCycle implements AppProvider
{
    ......
    protected void doStart() throws Exception
    {
        ......
        File scandir = _monitoredDir.getFile();
        Log.info("Deployment monitor " + scandir + " at interval " + _scanInterval);
        _scanner = new Scanner();
        _scanner.setScanDirs(Collections.singletonList(scandir));//设置要扫描的目录
        _scanner.setScanInterval(_scanInterval);//设置扫描间隔时间
        _scanner.setRecursive(_recursive);
        _scanner.setFilenameFilter(_filenameFilter);//设置文件/目录过滤器
        _scanner.setReportDirs(true);
        _scanner.addListener(_scannerListener);//设置扫描监听器,当文件/目录被新增/删除/修改时触发
        _scanner.start();
    }

    ......

    //应用被修改时触发的动作
    protected void fileChanged(String filename) throws Exception
    {
        if (Log.isDebugEnabled()) Log.debug("changed ",filename);
        App app = _appMap.remove(filename);//先删除
        if (app != null)
        {
            _deploymentManager.removeApp(app);
        }
        app = ScanningAppProvider.this.createApp(filename);//再重新部署
        if (app != null)
        {
            _appMap.put(filename,app);
            _deploymentManager.addApp(app);
        }
    }
}

public class Scanner
{
    ......
    public synchronized void start ()
    {
        if (_running)
            return;

        _running = true;

        if (_reportExisting)
        {
            // if files exist at startup, report them
            scan();
        }
        else
        {
            //just register the list of existing files and only report changes
            scanFiles();
            _prevScan.putAll(_currentScan);
        }
        schedule();//启动定时器
    }

    ......

    public synchronized void scan ()
    {
        scanFiles();
        reportDifferences(_currentScan, _prevScan);//扫描出修改过的应用,包括 新增/删除/修改
        _prevScan.clear();
        _prevScan.putAll(_currentScan);//将当前结果存储起来,以和下一次扫描结果进行比对
    }
}


四、下一步:

jetty 实现 servlet 容器原理分析

jetty Continuation 实现原理分析

你可能感兴趣的:(eclipse,xml,应用服务器,servlet,嵌入式)