分布式架构学习之:017--FastDFS分布式文件系统的安装与使用(单节点)

67 篇文章 1 订阅
4 篇文章 0 订阅

FastDFS是一个轻量级的开源分布式文件系统
FastDFS主要解决了大容量的文件存储和高并发访问的问题,文件存取时实现了负载均衡
FastDFS实现了软件方式的RAID,可以使用廉价的IDE硬盘进行存储
支持存储服务器在线扩容
支持相同内容的文件只保存一份,节约磁盘空间
FastDFS只能通过Client API访问,不支持POSIX访问方式
FastDFS特别适合大中型网站使用,用来存储资源文件(如:图片、文档、音频、视频等等)


系统架构-上传文件流程图


 1. client询问tracker上传到的storage,不需要附加参数;
 2. tracker返回一台可用的storage;
 3. client直接和storage通讯完成文件上传。


系统架构-下载文件流程图


 1. client询问tracker下载文件的storage,参数为文件标识(组名和文件名);
 2. tracker返回一台可用的storage;
 3. client直接和storage通讯完成文件下载。


Tracker Server:跟踪服务器,主要做调度工作,在访问上起负载均衡的作用。记录storage server的状态,是连接Client和Storage server的枢纽。
 Storage Server:存储服务器,文件和meta data都保存到存储服务器上
 group:组,也可称为卷。同组内服务器上的文件是完全相同的
 文件标识:包括两部分:组名和文件名(包含路径)
 meta data:文件相关属性,键值对(Key Value Pair)方式,如:width=1024,heigth=768

同一组内的storage server之间是对等的,文件上传、删除等操作可以在任意一台storage server上进行;
文件同步只在同组内的storage server之间进行,采用push方式,即源服务器同步给目标服务器;
源头数据才需要同步,备份数据不需要再次同步,否则就构成环路了;
上述第二条规则有个例外,就是新增加一台storage server时,由已有的一台storage server将已有的所有数据(包括源头数据和备份数据)同步给该新增服务器。

 协议包由两部分组成:header和body
 header共10字节,格式如下:
  8 bytes body length
  1 byte command
  1 byte status
 body数据包格式由取决于具体的命令,body可以为空


运行时目录结构-tracker server


运行时目录结构-storage server




FastDFS中文:http://www.csource.org/
FastDFS英文:http://code.google.com/p/fastdfs/


FastDFS 分布式文件系统的安装与使用(单节点)

跟踪服务器:192.168.4.121 (edu-dfs-tracker-01)

存储服务器:192.168.4.125 (edu-dfs-storage-01)

环境:CentOS 6.6

用户:root

数据目录:/fastdfs 注:数据目录按你的数据盘挂载路径而定

安装包:

FastDFS v5.05

libfastcommon-master.zip(是从 FastDFS  FastDHT 中提取出来的公共 C 函数库) 

fastdfs-nginx-module_v1.16.tar.gz

nginx-1.6.2.tar.gz fastdfs_client_java._v1.25.tar.gz

源码地址:https://github.com/happyfish100/

下载地址:http://sourceforge.net/projects/fastdfs/files/

官方论坛:http://bbs.chinaunix.net/forum-240-1.html

一、所有跟踪服务器和存储服务器均执行如下操作

1、编译和安装所需的依赖包:

# yum install make cmake gcc gcc-c++

2、安装 libfastcommon:

(1)上传或下载 libfastcommon-master.zip 到/usr/local/src 目录

(2)解压

cd /usr/local/src/

unzip libfastcommon-master.zip

cd libfastcommon-master

(3) 编译、安装

./make.sh

./make.sh install 

libfastcommon 默认安装到了

/usr/lib64/libfastcommon.so

/usr/lib64/libfdfsclient.so

(4)因为 FastDFS 主程序设置的 lib 目录是/usr/local/lib,所以需要创建软链接.

# ln -s /usr/lib64/libfastcommon.so /usr/local/lib/libfastcommon.so

ln -s /usr/lib64/libfastcommon.so /usr/lib/libfastcommon.so

ln -s /usr/lib64/libfdfsclient.so /usr/local/lib/libfdfsclient.so

ln -s /usr/lib64/libfdfsclient.so /usr/lib/libfdfsclient.so

3、安装 FastDFS

(1)上传或下载 FastDFS 源码包(FastDFS_v5.05.tar.gz)到 /usr/local/src 目录 

(2)解压

cd /usr/local/src/

tar -zxvf FastDFS_v5.05.tar.gz

cd FastDFS


(3)编译、安装(编译前要确保已经成功安装了 libfastcommon)

./make.sh

./make.sh install

采用默认安装的方式安装,安装后的相应文件与目录:

A、服务脚本在:

/etc/init.d/fdfs_storaged 

/etc/init.d/fdfs_tracker

B、配置文件在(样例配置文件):

/etc/fdfs/client.conf.sample

/etc/fdfs/storage.conf.sample

/etc/fdfs/tracker.conf.sample

C、命令工具在/usr/bin/目录下的:

fdfs_appender_test 

fdfs_appender_test1 

fdfs_append_file 

fdfs_crc32 

fdfs_delete_file 

fdfs_download_file 

fdfs_file_info 

fdfs_monitor 

fdfs_storaged

fdfs_test 

fdfs_test1 

fdfs_trackerd 

fdfs_upload_appender 

fdfs_upload_file 

stop.sh

restart.sh

(4)因为 FastDFS 服务脚本设置的 bin 目录是/usr/local/bin,但实际命令安装在/usr/bin,可以进入/user/bin 目录使用以下命令查看 fdfs 的相关命令:

cd /usr/bin/

ls | grep fdfs

 

 

因此需要修改 FastDFS 服务脚本中相应的命令路径,也就是把/etc/init.d/fdfs_storaged/etc/init.d/fdfs_tracker 两个脚本中的/usr/local/bin 修改成/usr/bin

# vi fdfs_trackerd

使用查找替换命令进统一修改:%s+/usr/local/bin+/usr/bin

# vi fdfs_storaged

使用查找替换命令进统一修改:%s+/usr/local/bin+/usr/bin 

二、配置 FastDFS 跟踪器(192.168.4.121)

1 复制 FastDFS 跟踪器样例配置文件,并重命名:

# cd /etc/fdfs/

# cp tracker.conf.sample tracker.conf

2 编辑跟踪器配置文件:

 # vi /etc/fdfs/tracker.conf 

修改的内容如下:

disabled=false

port=22122

base_path=/fastdfs/tracker

其它参数保留默认配置,具体配置解释请参考官方文档说明:

http://bbs.chinaunix.net/thread-1941456-1-1.html 

3 创建基础数据目录(参考基础目录 base_path 配置):

# mkdir -p /fastdfs/tracker

4 防火墙中打开跟踪器端口(默认为 22122):

# vi /etc/sysconfig/iptables

添加如下端口行:

-A INPUT -m state --state NEW -m tcp -p tcp --dport 22122 -j ACCEPT

重启防火墙:

# service iptables restart

5 启动 Tracker:

# /etc/init.d/fdfs_trackerd start

初次成功启动,会在/fastdfs/tracker 目录下创建 datalogs 两个目录)查看 FastDFS Tracker 是否已成功启动:

# ps -ef | grep fdfs

6 关闭 Tracker:

# /etc/init.d/fdfs_trackerd stop

7 设置 FastDFS 跟踪器开机启动:

# vi /etc/rc.d/rc.local

添加以下内容:

## FastDFS Tracker

/etc/init.d/fdfs_trackerd start

三、配置 FastDFS 存储(192.168.4.125)

1 复制 FastDFS 存储器样例配置文件,并重命名:

# cd /etc/fdfs/

# cp storage.conf.sample storage.conf

2 编辑存储器样例配置文件:

# vi /etc/fdfs/storage.conf

修改的内容如下: 

disabled=false 

port=23000 

base_path=/fastdfs/storage

store_path0=/fastdfs/storage 

tracker_server=192.168.4.121:22122 

http.server_port=8888

其它参数保留默认配置,具体配置解释请参考官方文档说明:

http://bbs.chinaunix.net/thread-1941456-1-1.html 

3 创建基础数据目录(参考基础目录 base_path 配置):

# mkdir -p /fastdfs/storage 

4 防火墙中打开存储器端口(默认为 23000):

# vi /etc/sysconfig/iptables

添加如下端口行:

-A INPUT -m state --state NEW -m tcp -p tcp --dport 23000 -j ACCEPT

重启防火墙:

# service iptables restart

5 启动 Storage:

# /etc/init.d/fdfs_storaged start

初次成功启动,会在/fastdfs/storage 目录下创建 datalogs 两个目录)查看 FastDFS Storage 是否已成功启动

# ps -ef | grep fdfs

6 关闭 Storage:

# /etc/init.d/fdfs_storaged stop

7 设置 FastDFS 存储器开机启动:

# vi /etc/rc.d/rc.local

添加:

## FastDFS Storage

/etc/init.d/fdfs_storaged start

四、文件上传测试(192.168.4.121)

1、修改 Tracker 服务器中的客户端配置文件:

cp /etc/fdfs/client.conf.sample /etc/fdfs/client.conf

vi /etc/fdfs/client.conf

base_path=/fastdfs/tracker

tracker_server=192.168.4.121:22122 

2、执行如下上传文件命令


 



千万不要使用kill -9来杀进程,可能导致binlog数据丢失



客户端Java代码测试
fdfs_client.conf
connect_timeout = 10
network_timeout = 30
charset = UTF-8
http.tracker_http_port = 8080
http.anti_steal_token = no
http.secret_key = FastDFS1234567890

tracker_server = 192.168.4.121:22122

封装的客户端代码
import java.io.ByteArrayInputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStream;

import org.apache.commons.lang3.StringUtils;
import org.apache.log4j.Logger;
import org.csource.common.NameValuePair;
import org.csource.fastdfs.ClientGlobal;
import org.csource.fastdfs.StorageClient1;
import org.csource.fastdfs.StorageServer;
import org.csource.fastdfs.TrackerClient;
import org.csource.fastdfs.TrackerServer;

/**
 * 
 * @描述: FastDFS分布式文件系统操作客户端 .
 * @作者: WuShuicheng .
 * @创建时间: 2015-3-29,下午8:13:49 .
 * @版本号: V1.0 .
 */
public class FastDFSClient {

	//private static final String CONF_FILENAME = Thread.currentThread().getContextClassLoader().getResource("").getPath() + "fdfs_client.conf";
	private static final String CONF_FILENAME = "src/main/resources/fdfs/fdfs_client.conf";
	private static StorageClient1 storageClient1 = null;

	private static Logger logger = Logger.getLogger(FastDFSClient.class);

	/**
	 * 只加载一次.
	 */
	static {
		try {
			logger.info("=== CONF_FILENAME:" + CONF_FILENAME);
			ClientGlobal.init(CONF_FILENAME);
			TrackerClient trackerClient = new TrackerClient(ClientGlobal.g_tracker_group);
			TrackerServer trackerServer = trackerClient.getConnection();
			if (trackerServer == null) {
				logger.error("getConnection return null");
			}
			StorageServer storageServer = trackerClient.getStoreStorage(trackerServer);
			if (storageServer == null) {
				logger.error("getStoreStorage return null");
			}
			storageClient1 = new StorageClient1(trackerServer, storageServer);
		} catch (Exception e) {
			logger.error(e);
		}
	}

	/**
	 * 
	 * @param file
	 *            文件
	 * @param fileName
	 *            文件名
	 * @return 返回Null则为失败
	 */
	public static String uploadFile(File file, String fileName) {
		FileInputStream fis = null;
		try {
			NameValuePair[] meta_list = null; // new NameValuePair[0];
			fis = new FileInputStream(file);
			byte[] file_buff = null;
			if (fis != null) {
				int len = fis.available();
				file_buff = new byte[len];
				fis.read(file_buff);
			}

			String fileid = storageClient1.upload_file1(file_buff, getFileExt(fileName), meta_list);
			return fileid;
		} catch (Exception ex) {
			logger.error(ex);
			return null;
		}finally{
			if (fis != null){
				try {
					fis.close();
				} catch (IOException e) {
					logger.error(e);
				}
			}
		}
	}

	/**
	 * 根据组名和远程文件名来删除一个文件
	 * 
	 * @param groupName
	 *            例如 "group1" 如果不指定该值,默认为group1
	 * @param fileName
	 *            例如"M00/00/00/wKgxgk5HbLvfP86RAAAAChd9X1Y736.jpg"
	 * @return 0为成功,非0为失败,具体为错误代码
	 */
	public static int deleteFile(String groupName, String fileName) {
		try {
			int result = storageClient1.delete_file(groupName == null ? "group1" : groupName, fileName);
			return result;
		} catch (Exception ex) {
			logger.error(ex);
			return 0;
		}
	}

	/**
	 * 根据fileId来删除一个文件(我们现在用的就是这样的方式,上传文件时直接将fileId保存在了数据库中)
	 * 
	 * @param fileId
	 *            file_id源码中的解释file_id the file id(including group name and filename);例如 group1/M00/00/00/ooYBAFM6MpmAHM91AAAEgdpiRC0012.xml
	 * @return 0为成功,非0为失败,具体为错误代码
	 */
	public static int deleteFile(String fileId) {
		try {
			int result = storageClient1.delete_file1(fileId);
			return result;
		} catch (Exception ex) {
			logger.error(ex);
			return 0;
		}
	}

	/**
	 * 修改一个已经存在的文件
	 * 
	 * @param oldFileId
	 *            原来旧文件的fileId, file_id源码中的解释file_id the file id(including group name and filename);例如 group1/M00/00/00/ooYBAFM6MpmAHM91AAAEgdpiRC0012.xml
	 * @param file
	 *            新文件
	 * @param filePath
	 *            新文件路径
	 * @return 返回空则为失败
	 */
	public static String modifyFile(String oldFileId, File file, String filePath) {
		String fileid = null;
		try {
			// 先上传
			fileid = uploadFile(file, filePath);
			if (fileid == null) {
				return null;
			}
			// 再删除
			int delResult = deleteFile(oldFileId);
			if (delResult != 0) {
				return null;
			}
		} catch (Exception ex) {
			logger.error(ex);
			return null;
		}
		return fileid;
	}

	/**
	 * 文件下载
	 * 
	 * @param fileId
	 * @return 返回一个流
	 */
	public static InputStream downloadFile(String fileId) {
		try {
			byte[] bytes = storageClient1.download_file1(fileId);
			InputStream inputStream = new ByteArrayInputStream(bytes);
			return inputStream;
		} catch (Exception ex) {
			logger.error(ex);
			return null;
		}
	}

	/**
	 * 获取文件后缀名(不带点).
	 * 
	 * @return 如:"jpg" or "".
	 */
	private static String getFileExt(String fileName) {
		if (StringUtils.isBlank(fileName) || !fileName.contains(".")) {
			return "";
		} else {
			return fileName.substring(fileName.lastIndexOf(".") + 1); // 不带最后的点
		}
	}
}

测试代码
import java.io.File;
import java.io.InputStream;

import org.apache.commons.io.FileUtils;

import wusc.edu.demo.fdfs.FastDFSClient;
import org.junit.Test; 

/**
 * 
 * @描述: FastDFS测试 .
 * @作者: WuShuicheng .
 * @创建时间: 2015-3-29,下午8:11:36 .
 * @版本号: V1.0 .
 */
public class FastDFSTest {
	
	/**
	 * 上传测试.
	 * @throws Exception
	 */
	@Test
public void upload() throws Exception {String filePath = "E:/WorkSpaceSpr10.6/edu-demo-fdfs/TestFile/DubboVideo.jpg";File file = new File(filePath);String fileId = FastDFSClient.uploadFile(file, filePath);System.out.println("Upload local file " + filePath + " ok, fileid=" + fileId);// fileId: group1/M00/00/00/wKgEfVUYPieAd6a0AAP3btxj__E335.jpg// url: http://192.168.4.125:8888/group1/M00/00/00/wKgEfVUYPieAd6a0AAP3btxj__E335.jpg}/** * 下载测试. * @throws Exception */
        @Test
public static void download() throws Exception {String fileId = "group1/M00/00/00/wKgEfVUYPieAd6a0AAP3btxj__E335.jpg";InputStream inputStream = FastDFSClient.downloadFile(fileId);File destFile = new File("E:/WorkSpaceSpr10.6/edu-demo-fdfs/TestFile/DownloadTest.jpg");FileUtils.copyInputStreamToFile(inputStream, destFile);}/** * 删除测试 * @throws Exception */
	@Test
public static void delete() throws Exception {String fileId = "group1/M00/00/00/wKgEfVUYPieAd6a0AAP3btxj__E335.jpg";int result = FastDFSClient.deleteFile(fileId);System.out.println(result == 0 ? "删除成功" : "删除失败:" + result);}/** * @param args * @throws Exception */
	@Test
public static void main(String[] args) throws Exception {//upload();//download();delete();}}



  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值