[mydocker]---一步步实现volume操作

1. 准备工作

1.1 准备环境

root@nicktming:~/go/src/github.com/nicktming/mydocker# git clone https://github.com/nicktming/mydocker.git
root@nicktming:~/go/src/github.com/nicktming/mydocker# git checkout code-4.2
root@nicktming:~/go/src/github.com/nicktming/mydocker# git checkout -b dev-4.3

1.2 本文最终效果

-------------------------------terminal 01--------------------------------
root@nicktming:/nicktming# ls
busybox.tar  volume
root@nicktming:/nicktming# tree volume/
volume/
`-- test01.txt

0 directories, 1 file
root@nicktming:/nicktming# cat volume/test01.txt 
volume
root@nicktming:/nicktming# 

// 启动容器
-------------------------------terminal 02--------------------------------
root@nicktming:~/go/src/github.com/nicktming/mydocker# pwd
/root/go/src/github.com/nicktming/mydocker
root@nicktming:~/go/src/github.com/nicktming/mydocker# git clone https://github.com/nicktming/mydocker.git
root@nicktming:~/go/src/github.com/nicktming/mydocker# git checkout code-4.3
root@nicktming:~/go/src/github.com/nicktming/mydocker# go build .
root@nicktming:~/go/src/github.com/nicktming/mydocker# ./mydocker run -it -v /nicktming/volume:/containerVolume -v /nicktming/volume01:/root/volume01 /bin/sh
2019/04/08 00:12:13 volume:[/nicktming/volume:/containerVolume /nicktming/volume01:/root/volume01]
2019/04/08 00:12:13 rootPath:
2019/04/08 00:12:13 rootPath is empaty, set cmd.Dir by default: /nicktming/mnt
2019/04/08 00:12:13 current path: /nicktming/mnt.
/ # ls -l
total 48
drwxr-xr-x    2 root     root         12288 Feb 14 18:58 bin
drwxr-xr-x    4 root     root          4096 Apr  7 16:12 containerVolume
drwxr-xr-x    4 root     root          4096 Mar 17 16:05 dev
drwxr-xr-x    3 root     root          4096 Mar 17 16:05 etc
drwxr-xr-x    2 nobody   nogroup       4096 Feb 14 18:58 home
dr-xr-xr-x  102 root     root             0 Apr  7 16:12 proc
drwx------    3 root     root          4096 Apr  7 16:12 root
drwxr-xr-x    2 root     root          4096 Mar 17 16:05 sys
drwxrwxrwt    2 root     root          4096 Feb 14 18:58 tmp
drwxr-xr-x    3 root     root          4096 Feb 14 18:58 usr
drwxr-xr-x    4 root     root          4096 Feb 14 18:58 var
/ # cat containerVolume/test01.txt 
volume
/ # ls -l /root/volume01/
total 0
/ # echo "volume01" > /root/volume01/test02.txt
/ # echo "volume again" >> /containerVolume/test01.txt  
/ # exit
root@nicktming:~/go/src/github.com/nicktming/mydocker# 

// 查看宿主机内容
-------------------------------terminal 01--------------------------------
root@nicktming:/nicktming# ls
busybox  busybox.tar  volume  volume01
root@nicktming:/nicktming# cat volume/
test01.txt    .wh..wh.aufs  .wh..wh.orph/ .wh..wh.plnk/ 
root@nicktming:/nicktming# cat volume/test01.txt 
volume
volume again
root@nicktming:/nicktming# tree volume01/
volume01/
`-- test02.txt

0 directories, 1 file
root@nicktming:/nicktming# cat volume01/test02.txt 
volume01
root@nicktming:/nicktming# 

2. 存在的问题

[mydocker]---一步步实现使用AUFS包装busybox 中在容器内增删改文件都不会保存, 那如果用户需要保存起来怎么办. 在docker-v参数可以把宿主机的目录挂到容器内, 因此本文将会实现该功能.

3. 实现-v参数 (volume)

3.1 aufs 命令行实现

在上文中已经实现了将镜像层和容器层挂载到某一个目录(比如mnt)中, 接下来看看如何把用户指定目录挂载到容器中, 其实只需要在mnt中挂载一个目录到用户指定的宿主机目录即可.

root@nicktming:/nicktming# pwd
/nicktming
root@nicktming:/nicktming# ls
busybox.tar
root@nicktming:/nicktming# mkdir -p busybox mnt volume writerLayer
root@nicktming:/nicktming# tar -xvf busybox.tar -C busybox/
root@nicktming:/nicktming# ls
busybox  busybox.tar  mnt  volume  writerLayer
root@nicktming:/nicktming# df -h
Filesystem      Size  Used Avail Use% Mounted on
/dev/vda1        50G  2.7G   45G   6% /
none            4.0K     0  4.0K   0% /sys/fs/cgroup
udev            487M   12K  487M   1% /dev
tmpfs           100M  364K  100M   1% /run
none            5.0M     0  5.0M   0% /run/lock
none            497M   24K  497M   1% /run/shm
none            100M     0  100M   0% /run/user

volume: 是用户需要挂载到容器中的宿主机目录. 类似于-v /nicktming/volume:/containerVolume.
mnt: 是容器可以看到的目录, 也就是镜像层和容器层挂载的目录.
busybox: 镜像层
writerLayer: 容器层

执行两个mount操作,

root@nicktming:/nicktming# mount -t aufs -o dirs=/nicktming/writerLayer:/nicktming/busybox none /nicktming/mnt
// 宿主机挂载目录对应的容器目录
root@nicktming:/nicktming# mkdir -p mnt/containerVolume
root@nicktming:/nicktming# mount -t aufs -o dirs=/nicktming/volume none /nicktming/mnt/containerVolume
root@nicktming:/nicktming# df -h
Filesystem      Size  Used Avail Use% Mounted on
/dev/vda1        50G  2.7G   45G   6% /
none            4.0K     0  4.0K   0% /sys/fs/cgroup
udev            487M   12K  487M   1% /dev
tmpfs           100M  364K  100M   1% /run
none            5.0M     0  5.0M   0% /run/lock
none            497M   24K  497M   1% /run/shm
none            100M     0  100M   0% /run/user
none             50G  2.7G   45G   6% /nicktming/mnt
none             50G  2.7G   45G   6% /nicktming/mnt/containerVolume

操作文件进行测试, 往用户挂载的目录中写文件.

root@nicktming:/nicktming# echo "test01" > /nicktming/mnt/containerVolume/test01.txt
root@nicktming:/nicktming# tree volume/
volume/
`-- test01.txt

0 directories, 1 file
root@nicktming:/nicktming# cat volume/test01.txt 
test01
// 执行umount操作看看操作的数据是否保存到/nicktming/volume中
root@nicktming:/nicktming# umount /nicktming/mnt/containerVolume
root@nicktming:/nicktming# umount /nicktming/mnt
root@nicktming:/nicktming# 
root@nicktming:/nicktming# tree 
busybox/     busybox.tar  mnt/         volume/      writerLayer/ 
root@nicktming:/nicktming# tree mnt/
mnt/

0 directories, 0 files
root@nicktming:/nicktming# tree volume/
volume/
`-- test01.txt

0 directories, 1 file
root@nicktming:/nicktming# cat volume/test01.txt 
test01

3.2 实现volume操作

根据 3.1 aufs 命令行实现所示, 其实就是利用代码来实现上述命令.

3.2.1 增加-v参数

在**command/command.go中修改RunCommand如下:

var RunCommand = cli.Command{
    Name: "run",
    Flags: []cli.Flag {
        ...
        cli.StringFlag{
            Name: "v",
            Usage: "enable volume",
        },
    },
    Action: func(c *cli.Context) error {
        ...
        volume    := c.String("v")
        ...
        Run(command, tty, &cg, rootPath, volume)
        return nil
    },
}
3.2.1 增加处理volume的添加和删除方法
// 增加volume 并且mount操作
func CreateVolume(rootPath, volume string) error {
    if volume != "" {
        containerMntPath := rootPath + "/mnt"
        hostPath    := strings.Split(volume, ":")[0]
        exist, _ := PathExists(hostPath)
        if !exist {
            if err := os.Mkdir(hostPath, 0777); err != nil {
                log.Printf("mkdir %s err:%v\n", hostPath, err)
                return fmt.Errorf("mkdir %s err:%v\n", hostPath, err)
            }
        }
        mountPath   := strings.Split(volume, ":")[1]
        containerPath := containerMntPath + mountPath
        if err := os.Mkdir(containerPath, 0777); err != nil {
            log.Printf("mkdir %s err:%v\n", containerPath, err)
            return fmt.Errorf("mkdir %s err:%v\n", containerPath, err)
        }
        dirs := "dirs=" + hostPath
        if _, err := exec.Command("mount", "-t", "aufs", "-o", dirs, "none", containerPath).CombinedOutput(); err != nil {
            log.Printf("mount -t aufs -o %s none %s, err:%v\n", dirs, containerPath, err)
            return fmt.Errorf("mount -t aufs -o %s none %s, err:%v\n", dirs, containerPath, err)
        }
    }
    return nil
}
// 删除volume
func ClearVolume(rootPath, volume string)  {
    if volume != "" {
        containerMntPath   := rootPath + "/mnt"
        mountPath     := strings.Split(volume, ":")[1]
        containerPath := containerMntPath + mountPath
        if _, err := exec.Command("umount", "-f", containerPath).CombinedOutput(); err != nil {
            log.Printf("mount -f %s, err:%v\n", containerPath, err)
        }
        if err := os.RemoveAll(containerPath); err != nil {
            log.Printf("remove %s, err:%v\n", containerPath, err)
        }
    }
}
3.2.3 使用volume相关方法
// 使用CreateVolume 
func NewWorkDir(rootPath, volume string) error {
    if err := CreateContainerLayer(rootPath); err != nil {
        return fmt.Errorf("CreateContainerLayer(%s) error: %v.\n", rootPath, err)
    }
    if err := CreateMntPoint(rootPath); err != nil {
        return fmt.Errorf("CreateMntPoint(%s) error: %v.\n", rootPath, err)
    }
    if err := SetMountPoint(rootPath); err != nil {
        return fmt.Errorf("SetMountPoint(%s) error: %v.\n", rootPath, err)
    }
    if err := CreateVolume(rootPath, volume); err != nil {
        return fmt.Errorf("CreateVolume(%s, %s) error: %v.\n", rootPath, volume, err)
    }
    return nil
}
// 清理工作
func ClearWorkDir(rootPath, volume string)  {
    ClearVolume(rootPath, volume)
    ClearMountPoint(rootPath)
    ClearWriterLayer(rootPath)
}
3.2.4 测试
--------------------------------terminal 01-------------------------------
root@nicktming:/nicktming# pwd
/nicktming
root@nicktming:/nicktming# ls
busybox.tar

// 启动容器
--------------------------------terminal 02-------------------------------
root@nicktming:~/go/src/github.com/nicktming/mydocker# ./mydocker run -it -v /nicktming/volume:/containerVolume /bin/sh
2019/04/07 22:09:52 volume:/nicktming/volume:/containerVolume
2019/04/07 22:09:52 rootPath:
2019/04/07 22:09:52 rootPath is empaty, set cmd.Dir by default: /nicktming/mnt
2019/04/07 22:09:52 current path: /nicktming/mnt.
/ # ls
bin              dev              home             root             tmp              var
containerVolume  etc              proc             sys              usr
/ # ls -l
total 48
drwxr-xr-x    2 root     root         12288 Feb 14 18:58 bin
drwxr-xr-x    4 root     root          4096 Apr  7 14:09 containerVolume
drwxr-xr-x    4 root     root          4096 Mar 17 16:05 dev
drwxr-xr-x    3 root     root          4096 Mar 17 16:05 etc
drwxr-xr-x    2 nobody   nogroup       4096 Feb 14 18:58 home
dr-xr-xr-x  105 root     root             0 Apr  7 14:09 proc
drwx------    2 root     root          4096 Apr  7 14:10 root
drwxr-xr-x    2 root     root          4096 Mar 17 16:05 sys
drwxrwxrwt    2 root     root          4096 Feb 14 18:58 tmp
drwxr-xr-x    3 root     root          4096 Feb 14 18:58 usr
drwxr-xr-x    4 root     root          4096 Feb 14 18:58 var
/ # echo "test01" > containerVolume/test01.txt
/ # cat containerVolume/test01.txt 
test01

// 查看宿主机内容
--------------------------------terminal 01-------------------------------
root@nicktming:/nicktming# ls
busybox  busybox.tar  mnt  volume  writerLayer
root@nicktming:/nicktming# df -h
df: ‘/tmp/tmpMZmTH1’: No such file or directory
Filesystem      Size  Used Avail Use% Mounted on
/dev/vda1        50G  2.7G   45G   6% /
none            4.0K     0  4.0K   0% /sys/fs/cgroup
udev            487M   12K  487M   1% /dev
tmpfs           100M  372K  100M   1% /run
none            5.0M     0  5.0M   0% /run/lock
none            497M   24K  497M   1% /run/shm
none            100M     0  100M   0% /run/user
none             50G  2.7G   45G   6% /nicktming/mnt
none             50G  2.7G   45G   6% /nicktming/mnt/containerVolume
root@nicktming:/nicktming# 

// 退出容器
--------------------------------terminal 02-------------------------------
/ # exit
root@nicktming:~/go/src/github.com/nicktming/mydocker# 

// 查看宿主机内容
--------------------------------terminal 01-------------------------------
root@nicktming:/nicktming# ls
busybox  busybox.tar  volume
root@nicktming:/nicktming# tree volume/
volume/
`-- test01.txt

0 directories, 1 file
root@nicktming:/nicktming# cat volume/test01.txt 
test01
root@nicktming:/nicktming# 

可以看到宿主机的内容已经保存下来. 接下来利用该volume再次测试.

--------------------------------terminal 02-------------------------------
root@nicktming:~/go/src/github.com/nicktming/mydocker# ./mydocker run -it -v /nicktming/volume:/root/containerVolume /bin/sh
2019/04/07 22:20:30 volume:/nicktming/volume:/root/containerVolume
2019/04/07 22:20:30 rootPath:
2019/04/07 22:20:30 rootPath is empaty, set cmd.Dir by default: /nicktming/mnt
2019/04/07 22:20:30 current path: /nicktming/mnt.
/ # ls -l
total 44
drwxr-xr-x    2 root     root         12288 Feb 14 18:58 bin
drwxr-xr-x    4 root     root          4096 Mar 17 16:05 dev
drwxr-xr-x    3 root     root          4096 Mar 17 16:05 etc
drwxr-xr-x    2 nobody   nogroup       4096 Feb 14 18:58 home
dr-xr-xr-x  100 root     root             0 Apr  7 14:20 proc
drwx------    3 root     root          4096 Apr  7 14:20 root
drwxr-xr-x    2 root     root          4096 Mar 17 16:05 sys
drwxrwxrwt    2 root     root          4096 Feb 14 18:58 tmp
drwxr-xr-x    3 root     root          4096 Feb 14 18:58 usr
drwxr-xr-x    4 root     root          4096 Feb 14 18:58 var
/ # ls -l /root
total 4
drwxr-xr-x    4 root     root          4096 Apr  7 14:20 containerVolume
/ # ls -l /root/containerVolume/
total 4
-rw-r--r--    1 root     root             7 Apr  7 14:10 test01.txt
/ # echo "\ntest01 again\n" >> /root/containerVolume/test01.txt 
/ # echo "test02" > /root/containerVolume/test02.txt 
/ # exit
root@nicktming:~/go/src/github.com/nicktming/mydocker# 

// 查看宿主机
--------------------------------terminal 01-------------------------------
root@nicktming:/nicktming# pwd
/nicktming
root@nicktming:/nicktming# ls
busybox  busybox.tar  volume
root@nicktming:/nicktming# tree volume/
volume/
|-- test01.txt
`-- test02.txt

0 directories, 2 files
root@nicktming:/nicktming# cat volume/test01.txt 
test01
\ntest01 again\n
root@nicktming:/nicktming# cat volume/test02.txt 
test02
root@nicktming:/nicktming# 

4. 实现多个-v参数

第3部分已经实现了单个-v参数的操作, 此处将实现多个-v操作.

4.1 修改RunCommand参数

var RunCommand = cli.Command{
    Name: "run",
    Flags: []cli.Flag {
        ...
        cli.StringSliceFlag{
            Name: "v",
            Usage: "enable volume",
        },
        /*
        cli.StringFlag{
            Name: "v",
            Usage: "enable volume",
        },
        */
    },
    Action: func(c *cli.Context) error {
        ...
        volumes    := c.StringSlice("v")
        ...
        Run(command, tty, &cg, rootPath, volumes)
        return nil
    },
}

4.2 修改NewWorkDir 和 ClearWorkDir

改变参数将volume string变为volumes []string外层加一个循环即可.

func NewWorkDir(rootPath string, volumes []string) error {
    if err := CreateContainerLayer(rootPath); err != nil {
        return fmt.Errorf("CreateContainerLayer(%s) error: %v.\n", rootPath, err)
    }
    if err := CreateMntPoint(rootPath); err != nil {
        return fmt.Errorf("CreateMntPoint(%s) error: %v.\n", rootPath, err)
    }
    if err := SetMountPoint(rootPath); err != nil {
        return fmt.Errorf("SetMountPoint(%s) error: %v.\n", rootPath, err)
    }
    for _, volume := range volumes {
        if err := CreateVolume(rootPath, volume); err != nil {
            return fmt.Errorf("CreateVolume(%s, %s) error: %v.\n", rootPath, volume, err)
        }
    }
    return nil
}

func ClearWorkDir(rootPath string, volumes []string)  {
    for _, volume := range volumes {
        ClearVolume(rootPath, volume)
    }
    ClearMountPoint(rootPath)
    ClearWriterLayer(rootPath)
}

4.2 修改Run方法

因为NewWorkDirClearWorkDir已经修改, 所以调用此两个方法的command/run.go中的Run方法

func Run(command string, tty bool, cg *cgroups.CroupManger, rootPath string, volumes []string)  {
...
    log.Printf("volume:%s\n", volumes)

    newRootPath := getRootPath(rootPath)
    cmd.Dir = newRootPath + "/busybox"
    if err := NewWorkDir(newRootPath, volumes); err == nil {
        cmd.Dir = newRootPath + "/mnt"
    }
    defer ClearWorkDir(newRootPath, volumes)
...
}

4.3 测试

------------------------------terminal 01---------------------------------
root@nicktming:/nicktming# pwd
/nicktming
root@nicktming:/nicktming# ls
busybox  busybox.tar  volume
root@nicktming:/nicktming# 

// 创建容器
------------------------------terminal 02---------------------------------
root@nicktming:~/go/src/github.com/nicktming/mydocker# ./mydocker run -it -v /nicktming/volume01:/containerVolume01 -v /nicktming/volume02:/containerVolume02 /bin/sh
2019/04/07 23:18:14 volume:[/nicktming/volume01:/containerVolume01 /nicktming/volume02:/containerVolume02]
2019/04/07 23:18:14 rootPath:
2019/04/07 23:18:14 rootPath is empaty, set cmd.Dir by default: /nicktming/mnt
2019/04/07 23:18:14 current path: /nicktming/mnt.
/ # ls -l
total 52
drwxr-xr-x    2 root     root         12288 Feb 14 18:58 bin
drwxr-xr-x    4 root     root          4096 Apr  7 15:18 containerVolume01
drwxr-xr-x    4 root     root          4096 Apr  7 15:18 containerVolume02
drwxr-xr-x    4 root     root          4096 Mar 17 16:05 dev
drwxr-xr-x    3 root     root          4096 Mar 17 16:05 etc
drwxr-xr-x    2 nobody   nogroup       4096 Feb 14 18:58 home
dr-xr-xr-x   98 root     root             0 Apr  7 15:18 proc
drwx------    2 root     root          4096 Apr  7 15:18 root
drwxr-xr-x    2 root     root          4096 Mar 17 16:05 sys
drwxrwxrwt    2 root     root          4096 Feb 14 18:58 tmp
drwxr-xr-x    3 root     root          4096 Feb 14 18:58 usr
drwxr-xr-x    4 root     root          4096 Feb 14 18:58 var
/ # echo "containerVolume01" > containerVolume01/test001.txt
/ # echo "containerVolume02" > containerVolume02/test002.txt
/ # exit
root@nicktming:~/go/src/github.com/nicktming/mydocker# 

// 查看宿主机的内容
------------------------------terminal 01---------------------------------
root@nicktming:/nicktming# pwd
/nicktming
root@nicktming:/nicktming# ls
busybox  busybox.tar  volume  volume01  volume02
root@nicktming:/nicktming# tree volume01
volume01
`-- test001.txt

0 directories, 1 file
root@nicktming:/nicktming# cat volume01/test001.txt 
containerVolume01
root@nicktming:/nicktming# tree volume02/
volume02/
`-- test002.txt

0 directories, 1 file
root@nicktming:/nicktming# cat volume02/test002.txt 
containerVolume02
root@nicktming:/nicktming# 

5. 时序图

code-4.3.png

6. 参考

1. 自己动手写docker.(基本参考此书,加入一些自己的理解,加深对docker的理解)

7. 全部内容

mydocker.png

1. [mydocker]---环境说明
2. [mydocker]---urfave cli 理解
3. [mydocker]---Linux Namespace
4. [mydocker]---Linux Cgroup
5. [mydocker]---构造容器01-实现run命令
6. [mydocker]---构造容器02-实现资源限制01
7. [mydocker]---构造容器02-实现资源限制02
8. [mydocker]---构造容器03-实现增加管道
9. [mydocker]---通过例子理解存储驱动AUFS
10. [mydocker]---通过例子理解chroot 和 pivot_root
11. [mydocker]---一步步实现使用busybox创建容器
12. [mydocker]---一步步实现使用AUFS包装busybox
13. [mydocker]---一步步实现volume操作
14. [mydocker]---实现保存镜像
15. [mydocker]---实现容器的后台运行
16. [mydocker]---实现查看运行中容器
17. [mydocker]---实现查看容器日志
18. [mydocker]---实现进入容器Namespace
19. [mydocker]---实现停止容器
20. [mydocker]---实现删除容器
21. [mydocker]---实现容器层隔离
22. [mydocker]---实现通过容器制作镜像
23. [mydocker]---实现cp操作
24. [mydocker]---实现容器指定环境变量
25. [mydocker]---网际协议IP
26. [mydocker]---网络虚拟设备veth bridge iptables
27. [mydocker]---docker的四种网络模型与原理实现(1)
28. [mydocker]---docker的四种网络模型与原理实现(2)
29. [mydocker]---容器地址分配
30. [mydocker]---网络net/netlink api 使用解析
31. [mydocker]---网络实现
32. [mydocker]---网络实现测试

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 202,009评论 5 474
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 84,808评论 2 378
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 148,891评论 0 335
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,283评论 1 272
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,285评论 5 363
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,409评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,809评论 3 393
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,487评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,680评论 1 295
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,499评论 2 318
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,548评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,268评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,815评论 3 304
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,872评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,102评论 1 258
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,683评论 2 348
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,253评论 2 341

推荐阅读更多精彩内容