Zookeeper_安全认证讲解

比如ACL的认证

Zookeeper_安全认证讲解_第1张图片

ACL的全称就是Access Control List,这个认证它是做什么事呢,你想修改这个节点,你可以加一个认证操作,认证一般有

这4种模式,IP模式,digest模式,world模式,super模式,其实没有太大的用,相当于别的客户端访问我zookeeper服务器的

时候,得有一个认证,才能允许你修改节点,要不然就不让你去修改,你自己有兴趣可以去看一看,咱们其实去运行一下

DEMO就行了

 

package com.learn.zookeeper.auth;

import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.atomic.AtomicInteger;

import org.apache.zookeeper.CreateMode;
import org.apache.zookeeper.WatchedEvent;
import org.apache.zookeeper.Watcher;
import org.apache.zookeeper.Watcher.Event.EventType;
import org.apache.zookeeper.Watcher.Event.KeeperState;
import org.apache.zookeeper.ZooDefs.Ids;
import org.apache.zookeeper.ZooKeeper;
import org.apache.zookeeper.data.ACL;
import org.apache.zookeeper.data.Stat;

/**
 * 
 * @author Leon.Sun
 *
 */
public class ZookeeperAuth implements Watcher {

	/** 连接地址 */
	final static String CONNECT_ADDR = "localhost:2181";
	/** 测试路径 */
	/**
	 * 这里有一个testAuth的路径
	 */
	final static String PATH = "/testAuth";
	/**
	 * 就是有两个路径
	 * 一个是父节点parent
	 * 还有一个是子节点delNode
	 * 
	 */
	final static String PATH_DEL = "/testAuth/delNode";
	/** 认证类型 */
	/**
	 * 认证类型就是用digest
	 * 这种类型也是最常用的
	 * 控制权限
	 * 首先把你这个认证进行两次的处理
	 * 第一次使用SHA-1
	 * 加密算法去加密
	 * BASE64编码
	 * 
	 */
	final static String authentication_type = "digest";
	/** 认证正确方法 */
	/**
	 * 第一次去连的时候需要输入一个字符串
	 * 以后机器连的时候必须使用一样的字符串才能修改
	 * 如果你想输入错误的不允许修改
	 * 其实就是一个非常easy的事
	 * 是从网上copy的一段代码
	 * 这不是我写的
	 * 但是这么做已经足够了
	 * 
	 */
	final static String correctAuthentication = "123456";
	/** 认证错误方法 */
	final static String badAuthentication = "654321";
	
	static ZooKeeper zk = null;
	/** 计时器 */
	AtomicInteger seq = new AtomicInteger();
	/** 标识 */
	private static final String LOG_PREFIX_OF_MAIN = "【Main】";
	
	private CountDownLatch connectedSemaphore = new CountDownLatch(1);
	
	@Override
	public void process(WatchedEvent event) {
		try {
			Thread.sleep(200);
		} catch (InterruptedException e) {
			e.printStackTrace();
		}
		if (event==null) {
			return;
		}
		// 连接状态
		KeeperState keeperState = event.getState();
		// 事件类型
		EventType eventType = event.getType();
		// 受影响的path
		String path = event.getPath();
		
		String logPrefix = "【Watcher-" + this.seq.incrementAndGet() + "】";

		System.out.println(logPrefix + "收到Watcher通知");
		System.out.println(logPrefix + "连接状态:\t" + keeperState.toString());
		System.out.println(logPrefix + "事件类型:\t" + eventType.toString());
		if (KeeperState.SyncConnected == keeperState) {
			// 成功连接上ZK服务器
			if (EventType.None == eventType) {
				System.out.println(logPrefix + "成功连接上ZK服务器");
				connectedSemaphore.countDown();
			} 
		} else if (KeeperState.Disconnected == keeperState) {
			System.out.println(logPrefix + "与ZK服务器断开连接");
		} else if (KeeperState.AuthFailed == keeperState) {
			System.out.println(logPrefix + "权限检查失败");
		} else if (KeeperState.Expired == keeperState) {
			System.out.println(logPrefix + "会话失效");
		}
		System.out.println("--------------------------------------------");
	}
	/**
	 * 创建ZK连接
	 * 
	 * createConnection的时候
	 * 第一次创建连接的时候
	 * 
	 * 
	 * @param connectString
	 *            ZK服务器地址列表
	 * @param sessionTimeout
	 *            Session超时时间
	 */
	public void createConnection(String connectString, int sessionTimeout) {
		this.releaseConnection();
		try {
			/**
			 * zookeeper加的东西是啥
			 * 肯定是this
			 * 就是一个Watcher
			 * 加了一个Watcher
			 * 
			 * 
			 */
			zk = new ZooKeeper(connectString, sessionTimeout, this);
			//添加节点授权
			/**
			 * 就是多了一个这个方法
			 * 这是zookeeper原生带的一个方法
			 * 就是提供认证
			 * 就是你第一次连接zookeeper的时候
			 * 你得加一个认证
			 * 认证的type就是上面的digest
			 * 这个字符串是固定的
			 * 当然你也可以用其他的
			 * 一般这种方式是最常用的
			 * 把这句话写到了这一行了
			 * 第一次连接的时候加上这个字符串
			 * authentication_type这个相当于一个key了
			 * correctAuthentication.getBytes()这个是value
			 * 这个value是自己随便写的
			 * 我这写的是123456
			 * 也就是我以后再连接zookeeper的时候
			 * 就是对这个节点进行增删改查的时候
			 * 我只能是加上这个认证才能去修改
			 * 要不然是不能去修改的
			 * 没有认证是不好使的
			 * 说白了就是这么一个简单的东西
			 * 其他的方式其实也都差不多
			 * 通过IP指定一台机器
			 * 只能这一台机器进行修改
			 * 其他机器是不允许修改
			 * 就是之前说的IP的模式
			 * 还有是world的模式
			 * 这个是开放的
			 * 这是一种特殊的digest
			 * 他仅仅是一个标识而已
			 * 还有一种是super
			 * 超级用户模式
			 * 基本上最常用的也就是这个了
			 * IP就是指定了一台机器
			 * 或者是按网段进行分配
			 * 192.168.1.*
			 * 简单的说完这些事情
			 * 咱们看一下吧
			 * 连完zookeeper以后
			 * 创建完connection之后
			 * 
			 * 自己定义的时候要加一个认证
			 * key是不变的
			 * value就相当于你的密码一样的
			 * 你可以自己随便定义一个很复杂的密码
			 * zk这个对象创建所有的节点
			 * 由于zk加认证创建的节点
			 * 
			 */
			zk.addAuthInfo(authentication_type,correctAuthentication.getBytes());
			System.out.println(LOG_PREFIX_OF_MAIN + "开始连接ZK服务器");
			//倒数等待
			connectedSemaphore.await();
		} catch (Exception e) {
			e.printStackTrace();
		}
	}
	
	/**
	 * 关闭ZK连接
	 */
	public void releaseConnection() {
		if (this.zk!=null) {
			try {
				this.zk.close();
			} catch (InterruptedException e) {
			}
		}
	}
	
	/**
	 * 
	 * 方法名称:测试函数
* 概要说明:测试认证
* @param args * @throws Exception */ public static void main(String[] args) throws Exception { ZookeeperAuth testAuth = new ZookeeperAuth(); testAuth.createConnection(CONNECT_ADDR,2000); List acls = new ArrayList(1); for (ACL ids_acl : Ids.CREATOR_ALL_ACL) { acls.add(ids_acl); } try { /** * 就是创建的时候要加上acls认证 * * 这个认证是zookeeper给你提供的 * for循环加到list里就行了 * 你加上这个认证以后 * 那这个节点需要认证才能去删除 * 我这边连zookeeper是可以连上 * 我可以连上zookeeper * 我要修改你刚创建的path节点 * 你要有一个认证才行 * 没有认证是不行的 * 要理解这个事情 * create(String path, byte[] data, List acl, CreateMode createMode) * create的时候要传一个list * 或者你穿其他的也行 * 创建节点的时候需要指定一种方式 * 你千万要注意一点 * 你创建一个节点的是有认证的 * 我再创建一个新的节点 * 这个认证只是针对于这一个节点 * 我的zookeeper只要操作一个新的节点 * 我是可以不加任何认证的 * 那个认证只是针对于某一个节点 * 要明白这个事情 */ zk.create(PATH, "init content".getBytes(), acls, CreateMode.PERSISTENT); System.out.println("使用授权key:" + correctAuthentication + "创建节点:"+ PATH + ", 初始内容是: init content"); } catch (Exception e) { e.printStackTrace(); } try { zk.create(PATH_DEL, "will be deleted! ".getBytes(), acls, CreateMode.PERSISTENT); System.out.println("使用授权key:" + correctAuthentication + "创建节点:"+ PATH_DEL + ", 初始内容是: init content"); } catch (Exception e) { e.printStackTrace(); } // 获取数据 getDataByNoAuthentication(); getDataByBadAuthentication(); getDataByCorrectAuthentication(); // 更新数据 updateDataByNoAuthentication(); updateDataByBadAuthentication(); updateDataByCorrectAuthentication(); // 删除数据 deleteNodeByBadAuthentication(); deleteNodeByNoAuthentication(); deleteNodeByCorrectAuthentication(); // Thread.sleep(1000); deleteParent(); //释放连接 testAuth.releaseConnection(); } /** 获取数据:采用错误的密码 */ /** * 使用错的 * 错误的是什么意思 * */ static void getDataByBadAuthentication() { String prefix = "[使用错误的授权信息]"; try { /** * 我连zookeeper的时候用认证 * 但是认证不对啊 * 我加了一个认证 * 但是是一个错误的认证 */ ZooKeeper badzk = new ZooKeeper(CONNECT_ADDR, 2000, null); //授权 /** * 我加的是一个bad的认证badAuthentication * 上面写的是什么 * 正确的是123456 * 我写的是654321 * 这个时候如果你认证写654321的时候 * 那就不行了 * 就认证失败了 * 打印的就是认证失败 * [使用错误的授权信息]更新失败,原因是:KeeperErrorCode = ConnectionLoss for /testAuth * 使用的是error * 为什么啊 * 如果这个改成正确的呢 * badAuthentication * 不就是123456吗 * */ badzk.addAuthInfo(authentication_type,badAuthentication.getBytes()); // badzk.addAuthInfo(authentication_type,"123456".getBytes()); Thread.sleep(2000); System.out.println(prefix + "获取数据:" + PATH); System.out.println(prefix + "成功获取数据:" + badzk.getData(PATH, false, null)); } catch (Exception e) { System.err.println(prefix + "获取数据失败,原因:" + e.getMessage()); } } /** 获取数据:不采用密码 */ /** * 没有认证的情况 * */ static void getDataByNoAuthentication() { String prefix = "[不使用任何授权信息]"; try { System.out.println(prefix + "获取数据:" + PATH); /** * 直接new一个ZooKeeper去连接 * 拿到这个nozk去get * */ ZooKeeper nozk = new ZooKeeper(CONNECT_ADDR, 2000, null); Thread.sleep(2000); /** * 这个是不允许的 * */ System.out.println(prefix + "成功获取数据:" + nozk.getData(PATH, false, null)); } catch (Exception e) { /** * 会产生一个失败的信息 * 就是会打印error这句话 * */ System.err.println(prefix + "获取数据失败,原因:" + e.getMessage()); } } /** 采用正确的密码 */ /** * 只有正确的方式 * 它是可以去获取的 * */ static void getDataByCorrectAuthentication() { String prefix = "[使用正确的授权信息]"; try { System.out.println(prefix + "获取数据:" + PATH); System.out.println(prefix + "成功获取数据:" + zk.getData(PATH, false, null)); } catch (Exception e) { System.out.println(prefix + "获取数据失败,原因:" + e.getMessage()); } } /** * 更新数据:不采用密码 */ static void updateDataByNoAuthentication() { String prefix = "[不使用任何授权信息]"; System.out.println(prefix + "更新数据: " + PATH); try { ZooKeeper nozk = new ZooKeeper(CONNECT_ADDR, 2000, null); Thread.sleep(2000); Stat stat = nozk.exists(PATH, false); if (stat!=null) { nozk.setData(PATH, prefix.getBytes(), -1); System.out.println(prefix + "更新成功"); } } catch (Exception e) { System.err.println(prefix + "更新失败,原因是:" + e.getMessage()); } } /** * 更新数据:采用错误的密码 */ static void updateDataByBadAuthentication() { String prefix = "[使用错误的授权信息]"; System.out.println(prefix + "更新数据:" + PATH); try { ZooKeeper badzk = new ZooKeeper(CONNECT_ADDR, 2000, null); //授权 badzk.addAuthInfo(authentication_type,badAuthentication.getBytes()); Thread.sleep(2000); Stat stat = badzk.exists(PATH, false); if (stat!=null) { badzk.setData(PATH, prefix.getBytes(), -1); System.out.println(prefix + "更新成功"); } } catch (Exception e) { System.err.println(prefix + "更新失败,原因是:" + e.getMessage()); } } /** * 更新数据:采用正确的密码 */ static void updateDataByCorrectAuthentication() { String prefix = "[使用正确的授权信息]"; System.out.println(prefix + "更新数据:" + PATH); try { Stat stat = zk.exists(PATH, false); if (stat!=null) { zk.setData(PATH, prefix.getBytes(), -1); System.out.println(prefix + "更新成功"); } } catch (Exception e) { System.err.println(prefix + "更新失败,原因是:" + e.getMessage()); } } /** * 不使用密码 删除节点 */ static void deleteNodeByNoAuthentication() throws Exception { String prefix = "[不使用任何授权信息]"; try { System.out.println(prefix + "删除节点:" + PATH_DEL); ZooKeeper nozk = new ZooKeeper(CONNECT_ADDR, 2000, null); Thread.sleep(2000); Stat stat = nozk.exists(PATH_DEL, false); if (stat!=null) { nozk.delete(PATH_DEL,-1); System.out.println(prefix + "删除成功"); } } catch (Exception e) { System.err.println(prefix + "删除失败,原因是:" + e.getMessage()); } } /** * 采用错误的密码删除节点 */ static void deleteNodeByBadAuthentication() throws Exception { String prefix = "[使用错误的授权信息]"; try { System.out.println(prefix + "删除节点:" + PATH_DEL); ZooKeeper badzk = new ZooKeeper(CONNECT_ADDR, 2000, null); //授权 badzk.addAuthInfo(authentication_type,badAuthentication.getBytes()); Thread.sleep(2000); Stat stat = badzk.exists(PATH_DEL, false); if (stat!=null) { badzk.delete(PATH_DEL, -1); System.out.println(prefix + "删除成功"); } } catch (Exception e) { System.err.println(prefix + "删除失败,原因是:" + e.getMessage()); } } /** * 使用正确的密码删除节点 */ static void deleteNodeByCorrectAuthentication() throws Exception { String prefix = "[使用正确的授权信息]"; try { System.out.println(prefix + "删除节点:" + PATH_DEL); Stat stat = zk.exists(PATH_DEL, false); if (stat!=null) { zk.delete(PATH_DEL, -1); System.out.println(prefix + "删除成功"); } } catch (Exception e) { System.out.println(prefix + "删除失败,原因是:" + e.getMessage()); } } /** * 使用正确的密码删除节点 */ static void deleteParent() throws Exception { try { Stat stat = zk.exists(PATH_DEL, false); if (stat == null) { zk.delete(PATH, -1); } } catch (Exception e) { e.printStackTrace(); } } }

 

你可能感兴趣的:(Zookeeper_安全认证讲解)