`
luhongjun
  • 浏览: 13780 次
  • 性别: Icon_minigender_1
  • 来自: 十堰
社区版块
存档分类
最新评论

OSGi系列 - 《OSGi实战》之用户验证

    博客分类:
  • osgi
 
阅读更多

 

这两天看BlueDavy写的一篇OSGi入门的文章《OSGi实战》,在道客巴巴有这篇文章的链接。

 

在这篇文章里面有这样一个例子:一个网站,有一个验证用户名和密码的页面(UserValidatorWebBundle)。验证的方式有三种:LDAP验证(LDAPValidatorBundle)﹑数据库验证(DBValidatorBundle)和配置文件验证(ConfigValidatorBundle)。在OSGi框架来讲,这分别是四个Bundle,UserValidatorWebBundle提供前端Web访问服务,执行时需要调用到三个验证Bundle中某一个提供的验证服务。三个验证Bundle在实际环境下可以通过启动其中一个﹑停止其它两个的方法实现动态的改变验证方式。

 

另外,还有一个UserValidatorBundle,这个Bundle实际上只包含了一个验证服务的接口类,在LDAPValidatorBundle﹑DBValidatorBundle和ConfigValidatorBundle三个Bundle之间提供验证服务的公共定义。

 

特别说明,我的开发环境是Eclipse JaveEE Indigo 3.7.2。

 

实现UserValidatorBundle

UserValidatorBundle提供验证服务的接口类Validator。由于它不要注册任何的服务和监听器,因此这里我们不需要为该Bundle提供Activator实现。它的作用就是将验证接口Validator通在MANIFEST.MF的Export-Package导出去,然后供其它具体实现验证服务的Bundle使用。

 

Bundle的详细开发过程可以参考《OSGi系列 - 用Eclipse开发Bundle》,这里只对关键地方进行说明。

 

UserValidatorBundle的目录结构,注意该Bundle没有Activator类:

 

Validator.java文件内容:

package org.riawork.demo.service.user;

public interface Validator {
    public boolean validate(String username, String password) throws Exception;
}

 

打开MANIFEST.MF文件,在出现的元数据编辑页面中选择底部的Runtime标签页,然后点击Exported Packages旁边的Add...按钮,添加如下的内容:

这个操作实际上就是在MANIFEST.MF文件里面增加一条Export-Package项目:

Export-Package: org.riawork.demo.service.user

 

MANIFEST.MF文件完整的内容:

复制代码
Manifest-Version: 1.0
Bundle-ManifestVersion: 2
Bundle-Name: UserValidatorBundle
Bundle-SymbolicName: UserValidatorBundle
Bundle-Version: 1.0.0
Import-Package: org.osgi.framework;version="1.3.0"
Bundle-RequiredExecutionEnvironment: JavaSE-1.6
Export-Package: org.riawork.demo.service.user
复制代码

 

实现LDAPValidatorBundle

LDAPValidatorBundle提供LDAP验证方式。

 

LDAPValidatorBundle提供了一个LDAPValidator验证类,该类需要实现UserValidatorBundle里面的Valdiator接口,因此需要在MANIFEST.MF文件中通过Import-Package导入Validator接口所在的包org.riawork.demo.service.user。

 

LDAPValidatorBundle同时也提供了一个LDAPActivator类,用来管理Bundle的生命周期,在start()和stop()方法中实现LDAPValidator验证服务的注册和卸载。

 

作为演示用途,这里所有的验证方式都使用的是字符串硬编码的简单方式。

 

LDAPValidatorBundle的目录结构:

 

打开MANIFEST.MF文件,在出现的元数据编辑页面中选择底部的Dependencies标签页,然后点击Imported Packages旁边的Add...按钮,添加如下的内容:

这个操作实际上就是在MANIFEST.MF文件里面增加一条Import-Package项目:

Import-Package: org.osgi.framework;version="1.3.0",
 org.riawork.demo.service.user

 

MANIFEST.MF文件完整的内容:

复制代码
Manifest-Version: 1.0
Bundle-ManifestVersion: 2
Bundle-Name: LDAPValidatorBundle
Bundle-SymbolicName: LDAPValidatorBundle
Bundle-Version: 1.0.0
Bundle-Activator: org.riawork.demo.service.user.impl.LDAPActivator
Import-Package: org.osgi.framework;version="1.3.0",
 org.riawork.demo.service.user
Bundle-RequiredExecutionEnvironment: JavaSE-1.6
复制代码

 

LDAPValidator.java文件内容:

复制代码
package org.riawork.demo.service.user.impl;

import org.riawork.demo.service.user.Validator;

public class LDAPValidator implements Validator {

    public boolean validate(String username, String password) throws Exception {
        System.out.println("LDAPValidator.validate()方法被调用");
        
        if ("jerry".equals(username) && "jerry".equals(password)){
            return true;
        }
        
        return false;
    }

}
复制代码

 

LDAPActivator.java文件内容:

复制代码
package org.riawork.demo.service.user.impl;

import org.osgi.framework.BundleActivator;
import org.osgi.framework.BundleContext;
import org.osgi.framework.ServiceRegistration;
import org.riawork.demo.service.user.Validator;

public class LDAPActivator implements BundleActivator {

    private ServiceRegistration serviceReg = null;

    public void start(BundleContext bundleContext) throws Exception {
        System.out.println("LDAPValidator服务被注册");
        
        serviceReg = bundleContext.registerService(Validator.class.getName(), new LDAPValidator(), null);
    }

    public void stop(BundleContext bundleContext) throws Exception {
        if(serviceReg != null)
            serviceReg.unregister();
        
        System.out.println("LDAPValidator服务被卸载");
    }
}
复制代码

 

实现DBValidatorBundle

DBValidatorBundle提供数据库验证方式,该Bundle的实现方式完全同LDAPValidatorBundle。

 

DBValidatorBundle的目录结构:

 

MANIFEST.MF文件内容:

复制代码
Manifest-Version: 1.0
Bundle-ManifestVersion: 2
Bundle-Name: DBValidatorBundle
Bundle-SymbolicName: DBValidatorBundle
Bundle-Version: 1.0.0
Bundle-Activator: org.riawork.demo.service.user.impl.DBActivator
Import-Package: org.osgi.framework;version="1.3.0",
 org.riawork.demo.service.user
Bundle-RequiredExecutionEnvironment: JavaSE-1.6
复制代码

 

DBValidator.java文件内容:

复制代码
package org.riawork.demo.service.user.impl;

import org.riawork.demo.service.user.Validator;

public class DBValidator implements Validator {

    public boolean validate(String username, String password) throws Exception {
        System.out.println("DBValidator.validate()方法被调用");
        
        if ("jerry".equals(username) && "jerry".equals(password)){
            return true;
        }
        
        return false;
    }
}
复制代码

 

DBActivator.java文件内容:

复制代码
package org.riawork.demo.service.user.impl;

import org.osgi.framework.BundleActivator;
import org.osgi.framework.BundleContext;
import org.osgi.framework.ServiceRegistration;
import org.riawork.demo.service.user.Validator;

public class DBActivator implements BundleActivator {

    private ServiceRegistration serviceReg = null;

    public void start(BundleContext bundleContext) throws Exception {
        System.out.println("DBValidator服务被注册");
        
        serviceReg = bundleContext.registerService(Validator.class.getName(), new DBValidator(), null);
    }

    public void stop(BundleContext bundleContext) throws Exception {
        if(serviceReg != null)
            serviceReg.unregister();
        
        System.out.println("DBValidator服务被卸载");
    }
}
复制代码

 

实现ConfigValidatorBundle

ConfigValidatorBundle提供配置人家验证方式,该Bundle的实现方式完全同LDAPValidatorBundle。

 

ConfigValidatorBundle的目录结构:

 

MANIFEST.MF文件内容:

复制代码
Manifest-Version: 1.0
Bundle-ManifestVersion: 2
Bundle-Name: ConfigValidatorBundle
Bundle-SymbolicName: ConfigValidatorBundle
Bundle-Version: 1.0.0
Bundle-Activator: org.riawork.demo.service.user.impl.ConfigActivator
Import-Package: org.osgi.framework;version="1.3.0",
 org.riawork.demo.service.user
Bundle-RequiredExecutionEnvironment: JavaSE-1.6
复制代码

 

ConfigValidator.java文件内容:

复制代码
package org.riawork.demo.service.user.impl;

import org.riawork.demo.service.user.Validator;

public class ConfigValidator implements Validator {

    public boolean validate(String username, String password) throws Exception {
        System.out.println("ConfigValidator.validate()方法被调用");
        
        if ("jerry".equals(username) && "jerry".equals(password)){
            return true;
        }
        
        return false;
    }
}
复制代码

 

ConfigActivator.java文件内容:

复制代码
package org.riawork.demo.service.user.impl;

import org.osgi.framework.BundleActivator;
import org.osgi.framework.BundleContext;
import org.osgi.framework.ServiceRegistration;
import org.riawork.demo.service.user.Validator;

public class ConfigActivator implements BundleActivator {

    private ServiceRegistration serviceReg = null;

    public void start(BundleContext bundleContext) throws Exception {
        System.out.println("ConfigValidator服务被注册");
        
        serviceReg = bundleContext.registerService(Validator.class.getName(), new ConfigValidator(), null);
    }

    public void stop(BundleContext bundleContext) throws Exception {
        if(serviceReg != null)
            serviceReg.unregister();
        
        System.out.println("ConfigValidator服务被卸载");
    }
}
复制代码

 

实现UserValidatorWebBundle

UserValidatorWebBundle提供使用者前端访问。该Bundle提供一个login.html页面,输入用户名和密码后,点击登录按钮,将数据提交到LoginServlet。LoginServlet检查,获取已经注册的Validator验证服务,调用validate()方法检查传过来的用户名和密码是否正确。

 

该Bundle使用到了Servlet API和HTTP服务,因此必需javax.servlet和org.eclipse.osgi.services两个额外的Bundle。

 

UserValidatorWebBundle的目录结构:

 

打开MANIFEST.MF文件,选择底部的Dependencies标签页,添加如下的内容:

这个操作实际上就是在MANIFEST.MF文件里面增加Import-Package项目和Require-Bundle项目:

Import-Package: org.osgi.framework;version="1.3.0",
 org.riawork.demo.service.user
Require-Bundle: javax.servlet;bundle-version="2.5.0",
 org.eclipse.osgi.services;bundle-version="3.3.0"

 

MANIFEST.MF文件完整的内容:

复制代码
Manifest-Version: 1.0
Bundle-ManifestVersion: 2
Bundle-Name: UserValidatorWebBundle
Bundle-SymbolicName: UserValidatorWebBundle
Bundle-Version: 1.0.0
Bundle-Activator: org.riawork.demo.web.Activator
Import-Package: org.osgi.framework;version="1.3.0",
 org.riawork.demo.service.user
Bundle-RequiredExecutionEnvironment: JavaSE-1.6
Require-Bundle: javax.servlet;bundle-version="2.5.0",
 org.eclipse.osgi.services;bundle-version="3.3.0"
复制代码

 

在src目录下新建page目录,然后在page目录下新增login.html文件。login.html文件内容:

复制代码
<HTML>
<HEAD>
<TITLE>OSGI Opendoc Demo</TITLE>
<meta http-equiv="Content-Type" content="text/html;charset=utf-8" /> 
<script>

    function login(){
        if(document.frmLogin.username.value==""){
            alert("必须输入用户名");
            document.frmLogin.username.focus();
            return;
        }
        if(document.frmLogin.userpass.value==""){
            alert("必须输入密码");
            document.frmLogin.userpass.focus();
            return;
        }
        document.frmLogin.submit();
    }

</script>
</HEAD>

<BODY>
<table style="width:45%;border:1px solid;height:100px">
    <form name=frmLogin method="post" action="/demo/login">
        <tr bgcolor="#C0C0C0">
            <td colspan="2" align="center">用户登录模块示例</td>
        </tr>
        <tr>
            <td width="30%">用户名</td>
            <td><input type="text" name="username" size="30"></td>
        </tr>
        <tr>
            <td>&nbsp;&nbsp;</td>
            <td><input type="password" name="userpass" size="33"></td>
        </tr>
        <tr bgcolor="#C0C0C0">
            <td colspan="2" align="center">
                <input type=button name=btnSubmit value="登录" onclick="login();">
            </td>
        </tr>
    </form>
</table>
</BODY>
</HTML>
复制代码

 

LoginServlet.java文件内容:

复制代码
package org.riawork.demo.web;

import java.io.IOException;

import javax.servlet.ServletOutputStream;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

import org.osgi.framework.BundleContext;
import org.osgi.framework.ServiceReference;
import org.riawork.demo.service.user.Validator;

public class LoginServlet extends HttpServlet {

    private static final long serialVersionUID = 1L;
    
    private BundleContext context;
    
    public LoginServlet(BundleContext context){
        this.context=context;
    }
    
    /** 
     * Web Post Method 
     */
    public void doPost(HttpServletRequest request,  HttpServletResponse response)
              throws IOException{
        doGet(request, response);
    }

    /** 
     * Web GET Method 
     */
    public void doGet(HttpServletRequest request,  HttpServletResponse response)
        throws IOException{
        String username=request.getParameter("username");
        String userpass=request.getParameter("userpass");
        response.setContentType("text/html");
        ServletOutputStream output=response.getOutputStream();
        ServiceReference serviceRef=context.getServiceReference(Validator.class.getName());
        Validator validator=(Validator) context.getService(serviceRef);
        if(validator==null){
            output.println("No usable validator service");
            output.close();
            return;
        }
        boolean result=false;
        try {
            result=validator.validate(username, userpass);
            if(result)
                output.println("Login success");
            else
                output.println("Login Fail,please check username and password");
            output.close();
            return;
        } 
        catch (Exception e) {
            output.println("Login error:");
            output.println(e.toString());
            return;
        }
    }

}
复制代码

 

Activator.java文件内容:

复制代码
package org.riawork.demo.web;

import javax.servlet.Servlet;

import org.osgi.framework.BundleActivator;
import org.osgi.framework.BundleContext;
import org.osgi.framework.ServiceEvent;
import org.osgi.framework.ServiceListener;
import org.osgi.framework.ServiceReference;
import org.osgi.service.http.HttpService;
import org.riawork.demo.web.LoginServlet;

public class Activator implements BundleActivator,ServiceListener{

    private BundleContext bc;
    private ServiceReference ref;
    private Servlet servlet;

    public void start(BundleContext context) throws Exception {
        bc=context;
        servlet=new LoginServlet(bc);
        registerServlet();
        context.addServiceListener(this, "(objectClass=" + HttpService.class.getName() + ")");
    }

    public void stop(BundleContext context) throws Exception {
        try {
            unregisterServlet();
        } catch (Throwable t) {
            t.printStackTrace();
        }

        servlet = null;
        bc = null;
        ref = null;
    }

    public void serviceChanged(ServiceEvent event) {
        switch (event.getType()){
            case ServiceEvent.REGISTERED:
                registerServlet();
                break;

            case ServiceEvent.UNREGISTERING:
                unregisterServlet();
                break;
        }
    }
    
    private void registerServlet() {
        if (ref == null)
            ref = bc.getServiceReference(HttpService.class.getName());
     
        if (ref != null) {
            try {
                HttpService http = (HttpService) bc.getService(ref);
                http.registerServlet("/demo/login", servlet, null, null);
                http.registerResources("/demo/page","page",null);
                System.out.println("已启动用户登录验证web模块,请通过/demo/page/login.htm访问");
            } 
            catch (Exception e) {
                e.printStackTrace();
            }
        }
    }
    
    private void unregisterServlet() {
        if (ref != null) {
            try {
                HttpService http = (HttpService) bc.getService(ref);
                http.unregister("/demo/login");
                http.unregister("/demo/page");
                System.out.println("已卸载用户登录验证web模块!");
            }
            catch(Exception e){
                e.printStackTrace();
            }
        }
    }
}
复制代码

 

创建OSGi Run Configuration

选择菜单Run >> Run Configurations...,在弹出的Run Configurations对话框左边找到OSGi Framework,在该项目上点击鼠标右键选择New菜单项,画面如下:

 

下表的Bundle在Run Configurations对话框中必须被选中:

  • UserValidatorBundle
  • LDAPValidatorBundle
  • DBValidatorBundle
  • ConfigValidatorBundle
  • UserValidatorWebBundle
  • javax.servlet
  • org.eclipse.osgi.services
  • org.eclipse.equinox.http.servlet
  • org.eclipse.equinox.http.jetty
  • org.mortbay.jetty.server
  • org.mortbay.jetty.util

 

配置完毕后,点击右下角的Run按钮,启动OSGi框架。如果一切正常,就会出现osgi>提示符,以及Bundle被启动时输出的提示信息:

 

从上图我们应该发现一个问题,实际上三个不同的验证方式都被启动了。

 

动态的切换验证方式

打开浏览器,输入http://localhost/demo/page/login.html,出现输入用户名和密码画面:

 

输入一个用户名和密码(正确的是jerry和jerry),点击登录按钮,浏览器网址转到了http://localhost/demo/login,这实际上就是我们注册的LoginServlet(参考UserValidatorWebBundle的Activator.java文件)。

 

请注意,这个时候我们查看OSGi控制台,有看到一条信息“ConfigValidator.validate()方法被调用”,说明当前使用的验证方式是配置文件验证。因为我们提供的三种验证服务都被启动了,换句话说,在OSGi框架中,Validator服务有三个提供者,在服务被调用时具体用的是哪一个提供者会由OSGi框架随机的决定。

 

停止Config和DB验证,只启动LDAP验证,然后再次提交http://localhost/demo/page/login.html,观察OSGi控制台,发现这次使用的是LDAP验证。

 

停止LDAP和Config验证,启动DB验证,观察OSGi控制台,是不是就是我们所期望的DB验证?

 

这篇文章实在是有点太长了,对于没耐心从头看到尾的朋友,下面提供了一个压缩文件下载链接,希望能帮助你省一点点时间。

 

分享到:
评论
发表评论

文章已被作者锁定,不允许评论。

相关推荐

    OSGI实战教程

    在OSGI实战教程中,首先需要了解OSGI(Open Services Gateway Initiative)是一个由众多IT公司共同制定的Java模块化标准规范,旨在实现软件组件的热插拔和服务动态管理。OSGI技术允许应用程序通过动态地安装、启动、...

    osgi 实战 pdf

    《OSGi实战》一书由BlueDavy撰写,是一份详尽的OSGi入门与实战指南,适合初学者和有经验的开发者深入了解OSGi框架及其应用。以下是对该书籍核心知识点的总结: ### OSGi简介 OSGi(Open Service Gateway Initiative...

    OSGI(实战中文版)

    文档还提供了一个可直接运行的环境,允许用户通过运行环境中的run.bat来访问用户登录验证模块。 文档的作者BlueDavy还讨论了OSGi框架的前瞻性和其带来的遐想,包括OSGi框架在未来技术发展中可能扮演的角色。文档中...

    osgi实战(详细文档)

    code.rar包含了基于Equinox的实战代码和使用EventAdminService的演示代码,而dist.rar则是一个可直接运行的环境,其中包含了运行OSGI应用所需的配置文件和脚本,解压后可以通过简单的操作来体验用户登录验证模块。...

    《OSGi实战》及其相关代码

    《OSGi实战》一书是Java开发者深入理解OSGi(Open Service Gateway Initiative)技术的重要参考资料。OSGi是一种模块化系统,它为Java应用程序提供了一种动态、可扩展的运行时环境。通过使用OSGi,开发者可以创建...

    osgi实战源码

    在《OSGi实战》这本书中,作者深入探讨了如何利用OSGi来构建可伸缩、灵活且可维护的Java应用程序。源码通常作为学习辅助资料,帮助读者更直观地理解书中的概念和技术。 OSGi的核心概念包括: 1. **模块系统**:...

    OSGI实战(PDF+源码+问题解决方法)

    首先,`OSGI实战.pdf`提供了对OSGI技术的全面介绍。这本书可能涵盖了以下几个方面的内容: 1. **OSGI基础**:解释OSGI的基本概念,如服务、模块(称为bundle)、生命周期(启动、暂停、停止、更新、删除等状态)...

    《osgi实战》.((美)Richard S.Hall)

    《OSGi实战》是美国作者Richard S. Hall所著的一本关于OSGi技术的实战指南。OSGi(Open Service Gateway Initiative)是一项面向Java的动态模块化系统规范,它允许在同一个应用程序中加载和运行多个版本的Java类库。...

    osgi doc

    本文档主要面向已经了解OSGi基本概念或已阅读过《OSGi实战》一书的读者。适合希望深入了解OSGi框架及其高级应用的技术人员。 ##### 1.2 编写目的 文档旨在通过具体案例深入讲解OSGi框架的应用技巧,帮助读者掌握...

    Spring_3.x企业应用开发实战 目录

    ### Spring 3.x 企业应用开发实战:关键知识点解析 #### 第1章 Spring概述 - **1.1 认识Spring** - **定义与作用**:Spring 是一个开源的轻量级 Java 开发框架,它主要的目标是解决企业级应用开发中的复杂性问题...

    ctk框架源码,以及ctkData资源

    1. **Qt库依赖**:CTK框架深度集成Qt库,利用其强大的图形用户界面设计能力,提供高效且用户友好的医疗应用UI。Qt库支持多平台运行,使得CTK框架具有良好的跨平台性。 2. **OSGi服务架构**:OSGi(Open Service ...

    MuleStudio用户手册.doc

    ### MuleStudio用户手册知识点概览 #### 一、Mule 概述 - **核心特点**:Mule 是一款以 Java 为核心的基础框架及整合平台,它支持超过30种传输协议,如 FTP、HTTP、SOAP 等,并且能够与 Spring、ActiveMQ 等流行...

    Spring3 Enterprise Recipes.pdf

    书中提供了一系列具体的优化策略,包括如何减少数据库访问延迟、如何利用缓存等技术来加速应用程序响应时间。 **案例研究与实践** - **实际项目示例**:为了加深理解,书中包含了多个实际项目的示例,涵盖了不同...

    struts2学习笔记

    - OSGi支持:使Struts2可以与其他OSGi容器集成。 8. **实际应用** - 项目结构:了解如何组织Maven或Gradle项目,以及目录结构的设置。 - 实战经验:如何在实际项目中使用Struts2,解决常见问题。 通过深入学习...

    蓝眼睛ServiceMix教程

    例如,设计一个简单的订单处理系统,该系统需要接收来自不同渠道的订单请求,然后经过一系列的处理步骤(如验证、库存检查等),最终将处理结果发送给相应的客户端。在这个过程中,可以利用ServiceMix的强大功能来...

    Learning Karaf Cellar

    ### 学习Karaf Cellar:构建与实施Apache Karaf OSGi容器的完整集群解决方案 #### 一、概述 《学习Karaf Cellar》这本书由Jean-Baptiste Onofré编写,旨在帮助读者深入了解如何在Apache Karaf OSGi容器上构建和...

    Eclipse RCP 插件开发指南

    JFace 向导是一系列对话框,引导用户完成一系列任务。向导通常用于执行复杂的操作,如创建新项目或配置设置等。 ##### Actions 动作(Action)是 Eclipse RCP 中用于封装命令逻辑的对象。通过将动作与菜单项、工具...

    基于Java的实例开发源码-Java源码Eclipse的Script插件 JSEditor.zip

    在JSEditor的源代码中,这个目录可能会包含一系列的Java类文件、配置文件以及必要的资源文件,这些文件共同构成了插件的核心功能。 在学习这个实例时,你可以深入了解以下知识点: 1. **Java编程基础**:理解并...

Global site tag (gtag.js) - Google Analytics