⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 ssh.c

📁 在LINUX下实现HA的源代码
💻 C
字号:
/* $Id: ssh.c,v 1.8.2.1 2004/04/20 08:50:06 alan Exp $ *//* * Stonith module for SSH Stonith device * * Copyright (c) 2001 SuSE Linux AG * * Authors: Joachim Gleissner <jg@suse.de>, Lars Marowsky-Br閑 <lmb@suse.de> * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. *  * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU * Lesser General Public License for more details. *  * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA * */#include <portability.h>#include <stdio.h>#include <stdlib.h>#include <unistd.h>#include <string.h>#include <errno.h>#include <syslog.h>#include <libintl.h>#include <sys/wait.h>#include <glib.h>#include <stonith/stonith.h>#define PIL_PLUGINTYPE          STONITH_TYPE#define PIL_PLUGINTYPE_S        STONITH_TYPE_S#define PIL_PLUGIN              ssh#define PIL_PLUGIN_S            "ssh"#define PIL_PLUGINLICENSE 	LICENSE_LGPL#define PIL_PLUGINLICENSEURL 	URL_LGPL#include <pils/plugin.h>/* * sshclose is called as part of unloading the ssh STONITH plugin. * If there was any global data allocated, or file descriptors opened, etc. * which is associated with the plugin, and not a single interface * in particular, here's our chance to clean it up. */static voidsshclosepi(PILPlugin*pi){}/* * sshcloseintf called as part of shutting down the ssh STONITH * interface.  If there was any global data allocated, or file descriptors * opened, etc.  which is associated with the ssh implementation, * here's our chance to clean it up. */static PIL_rcsshcloseintf(PILInterface* pi, void* pd){	return PIL_OK;}static void *		ssh_new(void);static void		ssh_destroy(Stonith *);static int		ssh_set_config_file(Stonith *, const char * cfgname);static int		ssh_set_config_info(Stonith *, const char * info);static const char *	ssh_getinfo(Stonith * s, int InfoType);static int		ssh_status(Stonith * );static int		ssh_reset_req(Stonith * s, int request, const char * host);static char **		ssh_hostlist(Stonith  *);static void		ssh_free_hostlist(char **);static struct stonith_ops sshOps ={	ssh_new,		/* Create new STONITH object	*/	ssh_destroy,		/* Destroy STONITH object	*/	ssh_set_config_file,	/* set configuration from file	*/	ssh_set_config_info,	/* Get configuration from file	*/	ssh_getinfo,		/* Return STONITH info string	*/	ssh_status,		/* Return STONITH device status	*/	ssh_reset_req,		/* Request a reset */	ssh_hostlist,		/* Return list of supported hosts */	ssh_free_hostlist	/* free above list */};PIL_PLUGIN_BOILERPLATE("1.0", Debug, sshclosepi);static const PILPluginImports*  PluginImports;static PILPlugin*               OurPlugin;static PILInterface*		OurInterface;static StonithImports*		OurImports;static void*			interfprivate;#define LOG		PluginImports->log#define MALLOC		PluginImports->alloc#define STRDUP  	PluginImports->mstrdup#define FREE		PluginImports->mfree#define EXPECT_TOK	OurImports->ExpectToken#define STARTPROC	OurImports->StartProcessPIL_rcPIL_PLUGIN_INIT(PILPlugin*us, const PILPluginImports* imports);PIL_rcPIL_PLUGIN_INIT(PILPlugin*us, const PILPluginImports* imports){	/* Force the compiler to do a little type checking */	(void)(PILPluginInitFun)PIL_PLUGIN_INIT;	PluginImports = imports;	OurPlugin = us;	/* Register ourself as a plugin */	imports->register_plugin(us, &OurPIExports);  	/*  Register our interface implementation */ 	return imports->register_interface(us, PIL_PLUGINTYPE_S	,	PIL_PLUGIN_S	,	&sshOps	,	sshcloseintf		/*close */	,	&OurInterface	,	(void*)&OurImports	,	&interfprivate); }#define	DEVICE	"SSH STONITH device"#define WHITESPACE	" \t\n\r\f"/* uncomment this if you have an ssh that can do what it claims#define SSH_COMMAND "ssh -q -x -o PasswordAuthentication=no StrictHostKeyChecking=no" *//* use this if you have the (broken) OpenSSH 2.1.1 */#define SSH_COMMAND "ssh -q -x -n -l root"/* We need to do a real hard reboot without syncing anything to simulate a * power cut.  * We have to do it in the background, otherwise this command will not * return. */#define REBOOT_COMMAND "nohup sh -c '(sleep 2; nohup /sbin/reboot -nf) </dev/null >/dev/null 2>&1' &"#undef REBOOT_COMMAND#define REBOOT_COMMAND	"echo 'sleep 2; /sbin/reboot -nf' | at now"/* *    SSH STONITH device * * I used the null device as template, so I guess there is missing * some functionality. * */struct sshDevice {  const char *	sshid;  char **		hostlist;  int		hostcount;};static const char * sshid = "SSHDevice-Stonith";static const char * NOTsshID = "SSH device has been destroyed";#define	ISSSHDEV(i)	(((i)!= NULL && (i)->pinfo != NULL)	\	&& ((struct sshDevice *)(i->pinfo))->sshid == sshid)#ifndef MALLOCT#	define     MALLOCT(t)      ((t *)(MALLOC(sizeof(t)))) #endif#define N_(text)	(text)#define _(text)		dgettext(ST_TEXTDOMAIN, text)static intssh_status(Stonith  *s){  if (!ISSSHDEV(s)) {    syslog(LOG_ERR, "invalid argument to SSH_status");    return(S_OOPS);  }  return S_OK;}/* *	Return the list of hosts configured for this SSH device */static char **ssh_hostlist(Stonith  *s){  int		numnames = 0;  char **		ret = NULL;  struct sshDevice*	sd;  int		j;  if (!ISSSHDEV(s)) {    syslog(LOG_ERR, "invalid argument to SSH_list_hosts");    return(NULL);  }  sd = (struct sshDevice*) s->pinfo;  if (sd->hostcount < 0) {    syslog(LOG_ERR	   ,	"unconfigured stonith object in SSH_list_hosts");    return(NULL);  }  numnames = sd->hostcount;  ret = (char **)MALLOC(numnames*sizeof(char*));  if (ret == NULL) {    syslog(LOG_ERR, "out of memory");    return ret;  }  memset(ret, 0, numnames*sizeof(char*));  for (j=0; j < numnames-1; ++j) {    ret[j] = STRDUP(sd->hostlist[j]);    if (ret[j] == NULL) {      ssh_free_hostlist(ret);      ret = NULL;      return ret;    }  }  return(ret);}static voidssh_free_hostlist (char ** hlist){  char **	hl = hlist;  if (hl == NULL) {    return;  }  while (*hl) {    FREE(*hl);    *hl = NULL;    ++hl;  }  FREE(hlist);  hlist = NULL;}static intWordCount(const char * s){  int	wc = 0;  if (!s) {    return wc;  }  do {    s += strspn(s, WHITESPACE);    if (*s)  {      ++wc;      s += strcspn(s, WHITESPACE);    }  }while (*s);  return(wc);}/* *	Parse the config information, and stash it away... */static intssh_parse_config_info(struct sshDevice* sd, const char * info){  char **			ret;  int			wc;  int			numnames;  const char *		s = info;  int			j;  if (sd->hostcount >= 0) {    return(S_OOPS);  }  wc = WordCount(info);  numnames = wc + 1;  ret = (char **)MALLOC(numnames*sizeof(char*));  if (ret == NULL) {    syslog(LOG_ERR, "out of memory");    return S_OOPS;  }  memset(ret, 0, numnames*sizeof(char*));  for (j=0; j < wc; ++j) {    s += strspn(s, WHITESPACE);    if (*s)  {      const char *	start = s;      s += strcspn(s, WHITESPACE);      ret[j] = MALLOC((1+(s-start))*sizeof(char));      if (ret[j] == NULL) {	ssh_free_hostlist(ret);	ret = NULL;	return S_OOPS;      }      strncpy(ret[j], start, (s-start));      g_strdown(ret[j]);    }  }  sd->hostlist = ret;  sd->hostcount = numnames;  return(S_OK);}/* *	Reset the given host on this Stonith device. */static intssh_reset_req(Stonith * s, int request, const char * host){  char cmd[4096];  if (!ISSSHDEV(s)) {    syslog(LOG_ERR, "invalid argument to %s", __FUNCTION__);    return(S_OOPS);  }  syslog(LOG_INFO, _("Host %s ssh-reset initiating"), host);  snprintf(cmd, 4096, "%s \"%s\" \"%s\"", SSH_COMMAND, host, REBOOT_COMMAND);    if (system(cmd) == 0)     return S_OK;  else {    syslog(LOG_ERR, "command %s failed", cmd);    return(S_RESETFAIL);  }}/* *	Parse the information in the given configuration file, *	and stash it away... */static intssh_set_config_file(Stonith* s, const char * configname){  FILE *	cfgfile;  char	line[256];  struct sshDevice*	sd;  if (!ISSSHDEV(s)) {    syslog(LOG_ERR, "invalid argument to SSH_set_configfile");    return(S_OOPS);  }  sd = (struct sshDevice*) s->pinfo;  if ((cfgfile = fopen(configname, "r")) == NULL)  {    syslog(LOG_ERR, "Cannot open %s", configname);    return(S_BADCONFIG);  }  while (fgets(line, sizeof(line), cfgfile) != NULL){    if (*line == '#' || *line == '\n' || *line == EOS) {      continue;    }    return(ssh_parse_config_info(sd, line));  }  return(S_BADCONFIG);}/* *	Parse the config information in the given string, and stash it away... */static intssh_set_config_info(Stonith* s, const char * info){  struct sshDevice* sd;  if (!ISSSHDEV(s)) {    syslog(LOG_ERR, "%s: invalid argument", __FUNCTION__);    return(S_OOPS);  }  sd = (struct sshDevice *)s->pinfo;  return(ssh_parse_config_info(sd, info));}static const char *ssh_getinfo(Stonith * s, int reqtype){  struct sshDevice* sd;  char *		ret;  if (!ISSSHDEV(s)) {    syslog(LOG_ERR, "SSH_idinfo: invalid argument");    return NULL;  }  /*   *	We look in the ST_TEXTDOMAIN catalog for our messages   */  sd = (struct sshDevice *)s->pinfo;  switch (reqtype) {  case ST_DEVICEID:    ret = _("ssh STONITH device");    break;  case ST_CONF_INFO_SYNTAX:    ret = _("hostname ...\n"	    "host names are white-space delimited.");    break;  case ST_CONF_FILE_SYNTAX:    ret = _("hostname...\n"	    "host names are white-space delimited.  "	    "All host names must be on one line.  "	    "Blank lines and lines beginning with # are ignored");    break;    case ST_DEVICEDESCR:		/* Description of device type */	ret = _("SSH-based Linux host reset\n"	"Fine for testing, but not suitable for production!");	break;  default:    ret = NULL;    break;  }  return ret;}/* *	SSH Stonith destructor... */static voidssh_destroy(Stonith *s){  struct sshDevice* sd;  if (!ISSSHDEV(s)) {    syslog(LOG_ERR, "%s: invalid argument", __FUNCTION__);    return;  }  sd = (struct sshDevice *)s->pinfo;  sd->sshid = NOTsshID;  if (sd->hostlist) {    ssh_free_hostlist(sd->hostlist);    sd->hostlist = NULL;  }  sd->hostcount = -1;  FREE(sd);}/* Create a new ssh Stonith device */static void *ssh_new(void){  struct sshDevice*	sd = MALLOCT(struct sshDevice);  if (sd == NULL) {    syslog(LOG_ERR, "out of memory");    return(NULL);  }  memset(sd, 0, sizeof(*sd));  sd->sshid = sshid;  sd->hostlist = NULL;  sd->hostcount = -1;  return((void *)sd);}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -