PXE高效批量网络装机

一,部署PXE远程安装服务

        1:搭建PXE远程安装服务器

                1.准备CentOS7安装源

  mkdir /mnt/centos
  mount -o loop /root/CentOS-7-x86_64-DVD.iso /mnt/centos
  
  • 复制安装源: 将必要的安装文件复制到TFTP服务器的指定目录,例如/var/lib/tftpboot/centos7:
  cp -r /mnt/centos/Packages /var/lib/tftpboot/centos7
  cp -r /mnt/centos/images/pxeboot/* /var/lib/tftpboot/centos7/pxeboot/
  

                2.安装并启用TFTP服务

  • 安装软件: 使用以下命令安装TFTP服务和xinetd:
  yum install tftp-server xinetd -y
  
  • 编辑TFTP配置: 打开/etc/xinetd.d/tftp,确保配置如下:
  service tftp
  {
      socket_type     = dgram
      protocol        = udp
      wait            = yes
      user            = root
      server          = /usr/sbin/in.tftpd
      server_args     = -s /var/lib/tftpboot
      disable         = no
      per_source      = 11
      cps             = 100 2
      flags           = IPv4
  }
  
  • 启动服务:
  systemctl start xinetd
  systemctl enable xinetd
  systemctl start tftp
  systemctl enable tftp
  

                3.准备Linux内核,初始化镜像文件

  • 直接从第1.1步复制的pxeboot目录中已有vmlinuzinitrd.img,无需额外操作。

                4.准备PXE引导程序

  • 安装syslinux:
  yum install syslinux -y
  
  • 复制引导文件:
  cp /usr/share/syslinux/pxelinux.0 /var/lib/tftpboot/
  

                5.安装并启用DHCP服务

  • 安装DHCP:
  yum install dhcp -y
  
  • 编辑DHCP配置: 编辑/etc/dhcp/dhcpd.conf,添加如下内容(记得替换<your_tftp_server_ip>为你的TFTP服务器IP地址):
  subnet <your_subnet> netmask <your_netmask> {
      range <dhcp_start_ip> <dhcp_end_ip>;
      option routers <your_router_ip>;
      option domain-name-servers <dns_server_ip>;
      
      # PXE Settings
      next-server <your_tftp_server_ip>;
      filename "pxelinux.0";
  }
  
  • 启动DHCP服务:
  systemctl start dhcpd
  systemctl enable dhcpd
  

                6.配置启动菜单文件

  • 创建菜单文件: 在/var/lib/tftpboot/pxelinux.cfg目录下创建default文件,内容如下:
  default centos7
  prompt 0
  timeout 60
  
  label centos7
      kernel centos7/pxeboot/vmlinuz
      append initrd=centos7/pxeboot/initrd.img inst.stage2=http://<your_http_server_ip>/centos7 images/pxeboot/ks.cfg
  

其中,http://<your_http_server_ip>/centos7应指向包含CentOS安装包的HTTP服务器目录,ks.cfg是Kickstart文件,用于自动化安装过程。

        2:验证PXE网络安装

  • 客户端设置: 确保客户端计算机的BIOS或UEFI设置中,网络启动(PXE Boot)被设置为优先启动项。
  • 测试安装: 重新启动客户端计算机,它应该会自动从网络启动并显示安装菜单。按照菜单提示进行,即可开始安装过程。

以上步骤完成后,你就可以通过PXE网络安装CentOS 7了。注意,实际操作时可能需要根据具体的网络环境和需求进行适当调整。

二,实现Kickstart无人值守安装

        1:准备安装应答文件

在 CentOS7系统中安装 system-config-kickstart 工具之后,即可通过图形化向导工具来配置安装应答文件。如果用户对自动应答文件的配置比较熟悉,也可以直接编辑 CentOS7 安装后自动创建的应答文件(rootanaconda-ks.cfg),根据需要适当修订后使用。

                1.配置安装应答参数

  • 编写Kickstart文件: Kickstart文件是一个文本文件,包含了安装过程中需要的所有交互式输入信息。以下是基本结构示例,你可以根据实际需求进行修改:

  #version=DEVEL
  install
  url --url=http://<your_http_server_ip>/centos7
  lang en_US.UTF-8
  keyboard us
  timezone America/New_York
  rootpw --iscrypted <encrypted_password>
  network --bootproto dhcp --device eth0
  firewall --disabled
  selinux --disabled
  reboot
  text
  autopart
  bootloader --location=mbr
  %packages
  @base
  %end
  %post
  # Custom post-installation scripting here, if needed
  %end
  
  • install: 表明这是一个安装任务。
  • url: 指定安装源的URL。
  • langkeyboardtimezone: 分别设定语言、键盘布局和时区。
  • rootpw: 设置root用户的密码,--iscrypted表示密码已经加密。
  • network: 配置网络设置,这里假设使用DHCP。
  • firewallselinux: 关闭防火墙和SELinux。
  • autopart: 自动分区。
  • %packages: 指定要安装的软件包组或单个包。
  • %post: 定义安装后执行的脚本。

                2.保存自动答应文件

  • 将上述内容保存为一个文本文件,通常命名为ks.cfg,并上传到HTTP服务器的某个目录下(例如之前提到的/centos7目录)。

        2:实现批量自动装机

                1.启用自动答应文件

  • 在PXE菜单配置文件(如/var/lib/tftpboot/pxelinux.cfg/default)中,确保Kickstart文件的引用路径正确无误。在之前的示例中,我们已经通过以下参数指定了Kickstart文件的路径:

  append initrd=... inst.stage2=http://<your_http_server_ip>/centos7 images/pxeboot/ks.cfg
  

这里inst.stage2后面跟的是Kickstart文件的HTTP地址。

                2.验证无人值守安装

  • 客户端验证: 确认客户端的BIOS设置为PXE优先启动后,重新启动客户端。系统应自动从网络启动,遵循Kickstart文件中的配置进行无人干预安装。
  • 监控日志: 在安装服务器上,可以通过系统日志(如/var/log/messages或/var/log/dhcpd.log)来监控PXE启动和安装过程中的错误信息。
  • 验证安装结果: 安装完成后,客户端应当自动重启并进入已安装好的CentOS 7系统,无需任何人工干预。登录系统检查是否所有预期的软件包、配置都已正确应用。
package com.hexiang.utils; /** * @(#)DateUtil.java * * * @author kidd * @version 1.00 2007/8/8 */ import java.util.*; import java.text.*; import java.sql.Timestamp; public class DateUtils { /** * 时间范围:年 */ public static final int YEAR = 1; /** * 时间范围:季度 */ public static final int QUARTER = 2; /** * 时间范围:月 */ public static final int MONTH = 3; /** * 时间范围:旬 */ public static final int TENDAYS = 4; /** * 时间范围:周 */ public static final int WEEK = 5; /** * 时间范围:日 */ public static final int DAY = 6; /* 基准时间 */ private Date fiducialDate = null; private Calendar cal = null; private DateUtils(Date fiducialDate) { if (fiducialDate != null) { this.fiducialDate = fiducialDate; } else { this.fiducialDate = new Date(System.currentTimeMillis()); } this.cal = Calendar.getInstance(); this.cal.setTime(this.fiducialDate); this.cal.set(Calendar.HOUR_OF_DAY, 0); this.cal.set(Calendar.MINUTE, 0); this.cal.set(Calendar.SECOND, 0); this.cal.set(Calendar.MILLISECOND, 0); this.fiducialDate = this.cal.getTime(); } /** * 获取DateHelper实例 * * @param fiducialDate * 基准时间 * @return Date */ public static DateUtils getInstance(Date fiducialDate) { return new DateUtils(fiducialDate); } /** * 获取DateHelper实例, 使用当前时间作为基准时间 * * @return Date */ public static DateUtils getInstance() { return new DateUtils(null); } /** * 获取年的第一天 * * @param offset * 偏移量 * @return Date */ public Date getFirstDayOfYear(int offset) { cal.setTime(this.fiducialDate); cal.set(Calendar.YEAR, cal.get(Calendar.YEAR) + offset); cal.set(Calendar.MONTH, Calendar.JANUARY); cal.set(Calendar.DAY_OF_MONTH, 1); return cal.getTime(); } /** * 获取年的最后一天 * * @param offset * 偏移量 * @return Date */ public Date getLastDayOfYear(int offset) { cal.setTime(this.fiducialDate); cal.set(Calendar.YEAR, cal.get(Calendar.YEAR) + offset); cal.set(Calendar.MONTH, Calendar.DECEMBER); cal.set(Calendar.DAY_OF_MONTH, 31); return cal.getTime(); } /** * 获取季度的第一天 * * @param offset * 偏移量 * @return Date */ public Date getFirstDayOfQuarter(int offset) { cal.setTime(this.fiducialDate); cal.add(Calendar.MONTH, offset * 3); int mon = cal.get(Calendar.MONTH); if (mon >= Calendar.JANUARY && mon = Calendar.APRIL && mon = Calendar.JULY && mon = Calendar.OCTOBER && mon = Calendar.JANUARY && mon = Calendar.APRIL && mon = Calendar.JULY && mon = Calendar.OCTOBER && mon = 21) { day = 21; } else if (day >= 11) { day = 11; } else { day = 1; } if (offset > 0) { day = day + 10 * offset; int monOffset = day / 30; day = day % 30; cal.add(Calendar.MONTH, monOffset); cal.set(Calendar.DAY_OF_MONTH, day); } else { int monOffset = 10 * offset / 30; int dayOffset = 10 * offset % 30; if ((day + dayOffset) > 0) { day = day + dayOffset; } else { monOffset = monOffset - 1; day = day - dayOffset - 10; } cal.add(Calendar.MONTH, monOffset); cal.set(Calendar.DAY_OF_MONTH, day); } return cal.getTime(); } /** * 获取旬的最后一天 * * @param offset * 偏移量 * @return Date */ public Date getLastDayOfTendays(int offset) { Date date = getFirstDayOfTendays(offset + 1); cal.setTime(date); cal.add(Calendar.DAY_OF_MONTH, -1); return cal.getTime(); } /** * 获取周的第一天(MONDAY) * * @param offset * 偏移量 * @return Date */ public Date getFirstDayOfWeek(int offset) { cal.setTime(this.fiducialDate); cal.add(Calendar.DAY_OF_MONTH, offset * 7); cal.set(Calendar.DAY_OF_WEEK, Calendar.MONDAY); return cal.getTime(); } /** * 获取周的最后一天(SUNDAY) * * @param offset * 偏移量 * @return Date */ public Date getLastDayOfWeek(int offset) { cal.setTime(this.fiducialDate); cal.add(Calendar.DAY_OF_MONTH, offset * 7); cal.set(Calendar.DAY_OF_WEEK, Calendar.MONDAY); cal.add(Calendar.DAY_OF_MONTH, 6); return cal.getTime(); } /** * 获取指定时间范围的第一天 * * @param dateRangeType * 时间范围类型 * @param offset * 偏移量 * @return Date */ public Date getFirstDate(int dateRangeType, int offset) { return null; } /** * 获取指定时间范围的最后一天 * * @param dateRangeType * 时间范围类型 * @param offset * 偏移量 * @return Date */ public Date getLastDate(int dateRangeType, int offset) { return null; } /** * 根据日历的规则,为基准时间添加指定日历字段的时间量 * * @param field * 日历字段, 使用Calendar类定义的日历字段常量 * @param offset * 偏移量 * @return Date */ public Date add(int field, int offset) { cal.setTime(this.fiducialDate); cal.add(field, offset); return cal.getTime(); } /** * 根据日历的规则,为基准时间添加指定日历字段的单个时间单元 * * @param field * 日历字段, 使用Calendar类定义的日历字段常量 * @param up * 指定日历字段的值的滚动方向。true:向上滚动 / false:向下滚动 * @return Date */ public Date roll(int field, boolean up) { cal.setTime(this.fiducialDate); cal.roll(field, up); return cal.getTime(); } /** * 把字符串转换为日期 * * @param dateStr * 日期字符串 * @param format * 日期格式 * @return Date */ public static Date strToDate(String dateStr, String format) { Date date = null; if (dateStr != null && (!dateStr.equals(""))) { DateFormat df = new SimpleDateFormat(format); try { date = df.parse(dateStr); } catch (ParseException e) { e.printStackTrace(); } } return date; } /** * 把字符串转换为日期日期的格式为yyyy-MM-dd HH:ss * * @param dateStr * 日期字符串 * @return Date */ public static Date strToDate(String dateStr) { Date date = null; if (dateStr != null && (!dateStr.equals(""))) { if (dateStr.matches("\\d{4}-\\d{1,2}-\\d{1,2}")) { dateStr = dateStr + " 00:00"; } else if (dateStr.matches("\\d{4}-\\d{1,2}-\\d{1,2} \\d{1,2}")) { dateStr = dateStr + ":00"; } else { System.out.println(dateStr + " 格式不正确"); return null; } DateFormat df = new SimpleDateFormat("yyyy-MM-dd HH:ss"); try { date = df.parse(dateStr); } catch (ParseException e) { e.printStackTrace(); } } return date; } /** * 把日期转换为字符串 * * @param date * 日期实例 * @param format * 日期格式 * @return Date */ public static String dateToStr(Date date, String format) { return (date == null) ? "" : new SimpleDateFormat(format).format(date); } public static String dateToStr(Date date) { return (date == null) ? "" : new SimpleDateFormat("yyyy-MM-dd hh:mm:ss").format(date); } /** * 取得当前日期 年-月-日 * * @return Date */ public static String getCurrentDate() { DateFormat f = new SimpleDateFormat("yyyy-MM-dd"); return f.format(Calendar.getInstance().getTime()); } public static void main(String[] args) { DateUtils dateHelper = DateUtils.getInstance(); /* Year */ for (int i = -5; i <= 5; i++) { System.out.println("FirstDayOfYear(" + i + ") = " + dateHelper.getFirstDayOfYear(i)); System.out.println("LastDayOfYear(" + i + ") = " + dateHelper.getLastDayOfYear(i)); } /* Quarter */ for (int i = -5; i <= 5; i++) { System.out.println("FirstDayOfQuarter(" + i + ") = " + dateHelper.getFirstDayOfQuarter(i)); System.out.println("LastDayOfQuarter(" + i + ") = " + dateHelper.getLastDayOfQuarter(i)); } /* Month */ for (int i = -5; i <= 5; i++) { System.out.println("FirstDayOfMonth(" + i + ") = " + dateHelper.getFirstDayOfMonth(i)); System.out.println("LastDayOfMonth(" + i + ") = " + dateHelper.getLastDayOfMonth(i)); } /* Week */ for (int i = -5; i <= 5; i++) { System.out.println("FirstDayOfWeek(" + i + ") = " + dateHelper.getFirstDayOfWeek(i)); System.out.println("LastDayOfWeek(" + i + ") = " + dateHelper.getLastDayOfWeek(i)); } /* Tendays */ for (int i = -5; i <= 5; i++) { System.out.println("FirstDayOfTendays(" + i + ") = " + dateHelper.getFirstDayOfTendays(i)); System.out.println("LastDayOfTendays(" + i + ") = " + dateHelper.getLastDayOfTendays(i)); } } /** * 取当前日期的字符串形式,"XXXX年XX月XX日" * * @return java.lang.String */ public static String getPrintDate() { String printDate = ""; Calendar calendar = Calendar.getInstance(); calendar.setTime(new Date()); printDate += calendar.get(Calendar.YEAR) + "年"; printDate += (calendar.get(Calendar.MONTH) + 1) + "月"; printDate += calendar.get(Calendar.DATE) + "日"; return printDate; } /** * 将指定的日期字符串转化为日期对象 * * @param dateStr * 日期字符串 * @return java.util.Date */ public static Date getDate(String dateStr, String format) { if (dateStr == null) { return new Date(); } if (format == null) { format = "yyyy-MM-dd"; } SimpleDateFormat sdf = new SimpleDateFormat(format); try { Date date = sdf.parse(dateStr); return date; } catch (Exception e) { return null; } } /** * 从指定Timestamp中得到相应的日期的字符串形式 日期"XXXX-XX-XX" * * @param dateTime * @return 、String */ public static String getDateFromDateTime(Timestamp dateTime) { SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd"); return sdf.format(dateTime).toString(); } /** * 得到当前时间 return java.sql.Timestamp * * @return Timestamp */ public static Timestamp getNowTimestamp() { long curTime = System.currentTimeMillis(); return new Timestamp(curTime); } }
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值