Teile diese Seite mit anderen

Lerne X in Y Minuten

Wobei X=ShutIt

ShutIt

ShuIt ist eine Shellautomationsframework, welches für eine einfache Handhabung entwickelt wurde.

Er ist ein Wrapper, der auf einem Python expect Klon (pexpect) basiert.

Es ist damit ein ‘expect ohne Schmerzen’.

Es ist verfügbar als pip install.

Hello World

Starten wir mit dem einfachsten Beispiel. Erstelle eine Datei names example.py

import shutit
session = shutit.create_session('bash')
session.send('echo Hello World', echo=True)

Führe es hiermit aus:

python example.py

gibt aus:

$ python example.py
echo "Hello World"
echo "Hello World"
Hello World
Ians-MacBook-Air.local:ORIGIN_ENV:RhuebR2T#

Das erste Argument zu ‘send’ ist der Befehl, den du ausführen möchtest. Das ‘echo’ Argument gibt die Terminalinteraktion aus. ShuIt ist standardmäßig leise.

‘Send’ kümmert sich um die nervige Arbeiten mit den Prompts und macht alles was du von ‘expect’ erwarten würdest.

Logge dich auf einen Server ein

Sagen wir du möchtest dich auf einen Server einloggen und einen Befehl ausführen. Ändere dafür example.py folgendermaßen:

import shutit
session = shutit.create_session('bash')
session.login('ssh [email protected]', user='du', password='meinpassword')
session.send('hostname', echo=True)
session.logout()

Dies erlaubt dir dich auf deinen Server einzuloggen (ersetze die Details mit deinen Eigenen) und das Programm gibt dir deinen Hostnamen aus.

$ python example.py
hostname
hostname
example.com
example.com:cgoIsdVv:heDa77HB#

Es ist klar das das nicht sicher ist. Stattdessen kann man Folgendes machen:

import shutit
session = shutit.create_session('bash')
password = session.get_input('', ispass=True)
session.login('ssh [email protected]', user='du', password=password)
session.send('hostname', echo=True)
session.logout()

Dies zwingt dich dein Passwort einzugeben:

$ python example.py
Input Secret:
hostname
hostname
example.com
example.com:cgoIsdVv:heDa77HB#

Die ‘login’ Methode übernimmt wieder das veränderte Prompt für den Login. Du übergibst ShutIt den User und das Passwort, falls es benötigt wird, mit den du dich einloggen möchtest. ShutIt übernimmt den Rest.

‘logout’ behandelt das Ende von ‘login’ und übernimmt alle Veränderungen des Prompts für dich.

Einloggen auf mehrere Server

Sagen wir, dass du eine Serverfarm mit zwei Servern hast und du dich in beide Server einloggen möchtest. Dafür musst du nur zwei Session und Logins erstellen und kannst dann Befehle schicken:

import shutit
session1 = shutit.create_session('bash')
session2 = shutit.create_session('bash')
password1 = session1.get_input('Password für server1', ispass=True)
password2 = session2.get_input('Password für server2', ispass=True)
session1.login('ssh [email protected]', user='du', password=password1)
session2.login('ssh [email protected]', user='du', password=password2)
session1.send('hostname', echo=True)
session2.send('hostname', echo=True)
session1.logout()
session2.logout()

Gibt aus:

$ python example.py
Password for server1
Input Secret:

Password for server2
Input Secret:
hostname
hostname
one.example.com
one.example.com:Fnh2pyFj:qkrsmUNs# hostname
hostname
two.example.com
two.example.com:Gl2lldEo:D3FavQjA#

Beispiel: Überwachen mehrerer Server

Wir können das obige Programm in ein einfaches Überwachungstool bringen indem wir Logik hinzufügen um die Ausgabe von einem Befehl zu betrachten.

import shutit
capacity_command="""df / | awk '{print $5}' | tail -1 | sed s/[^0-9]//"""
session1 = shutit.create_session('bash')
session2 = shutit.create_session('bash')
password1 = session.get_input('Passwort für Server1', ispass=True)
password2 = session.get_input('Passwort für Server2', ispass=True)
session1.login('ssh [email protected]', user='du', password=password1)
session2.login('ssh [email protected]', user='du', password=password2)
capacity = session1.send_and_get_output(capacity_command)
if int(capacity) < 10:
    print(kein Platz mehr auf Server1!')
capacity = session2.send_and_get_output(capacity_command)
if int(capacity) < 10:
    print(kein Platz mehr auf Server2!')
session1.logout()
session2.logout()

Hier kannst du die ‘send_and_get_output’ Methode verwenden um die Ausgabe von dem Kapazitätsbefehl (df) zu erhalten.

Es gibt elegantere Wege als oben (z.B. kannst du ein Dictionary verwenden um über die Server zu iterieren), aber es hängt and dir wie clever das Python sein muss.

kompliziertere IO - Expecting

Sagen wir du hast eine Interaktion mit einer interaktiven Kommandozeilenprogramm, die du automatisieren möchtest. Hier werden wir Telnet als triviales Beispiel verwenden:

import shutit
session = shutit.create_session('bash')
session.send('telnet', expect='elnet>', echo=True)
session.send('open google.com 80', expect='scape character', echo=True)
session.send('GET /', echo=True, check_exit=False)
session.logout()

Beachte das ‘expect’ Argument. Du brauchst nur ein Subset von Telnets Eingabeaufforderung um es abzugleichen und fortzufahren.

Beachte auch das neue Argument ‘check_exit’. Wir werden nachher nochmal darauf zurückkommen. Die Ausgabe von oben ist:

$ python example.py
telnet
telnet> open google.com 80
Trying 216.58.214.14...
Connected to google.com.
Escape character is '^]'.
GET /
HTTP/1.0 302 Found
Cache-Control: private
Content-Type: text/html; charset=UTF-8
Referrer-Policy: no-referrer
Location: http://www.google.co.uk/?gfe_rd=cr&ei=huczWcj3GfTW8gfq0paQDA
Content-Length: 261
Date: Sun, 04 Jun 2017 10:57:10 GMT

<HTML><HEAD><meta http-equiv="content-type" content="text/html;charset=utf-8">
<TITLE>302 Moved</TITLE></HEAD><BODY>
<H1>302 Moved</H1>
The document has moved
<A HREF="http://www.google.co.uk/?gfe_rd=cr&amp;ei=huczWcj3GfTW8gfq0paQDA">
here
</A>.
</BODY></HTML>
Connection closed by foreign host.

Nun zurück zu ‘check_exit=False’. Da das Telnet Programm einen Fehler mit Fehlercode (1) zurückgibt und wir nicht möchten das das Skript fehlschlägt kannst du ‘check_exit=False’ setzen und damit ShuIt wissen lassen, dass der Ausgabecode dich nicht interessiert.

Wenn du das Argument nicht mitgegeben hättest, dann hätte dir ShutIt ein interaktives Terminal zurückgegeben, falls es ein Terminal zum kommunizieren gibt. Dies nennt sich ein ‘Pause point’.

Pause Points

Du kannst jederzeit ‘pause point’ auslösen, wenn du Folgendes in deinem Skript aufrufst:

[...]
session.pause_point('Das ist ein pause point')
[...]

Danach kannst du das Skript fortführen, wenn du CTRL und ‘]’ zur selben Zeit drückst. Dies ist gut für Debugging: Füge ein Pause Point hinzu und schaue dich um. Danach kannst du das Programm weiter ausführen. Probiere folgendes aus:

import shutit
session = shutit.create_session('bash')
session.pause_point('Schaue dich um!')
session.send('echo "Hat dir der Pause point gefallen?"', echo=True)

Dies würde folgendes ausgeben:

$ python example.py
Schaue dich um!

Ians-Air.home:ORIGIN_ENV:I00LA1Mq#  bash
imiell@Ians-Air:/space/git/shutit   master +    
CTRL-] caught, continuing with run...
2017-06-05 15:12:33,577 INFO: Sending:  exit
2017-06-05 15:12:33,633 INFO: Output (squashed):  exitexitIans-Air.home:ORIGIN_ENV:I00LA1Mq#  [...]
echo "Hat dir der Pause point gefallen?"
echo "Hat dir der Pause point gefallen?"
Hat dir der Pause point gefallen?
Ians-Air.home:ORIGIN_ENV:I00LA1Mq#

noch kompliziertere IO - Hintergrund

Kehren wir zu unseren Beispiel mit dem Überwachen von mehreren Servern zurück. Stellen wir uns vor, dass wir eine langlaufende Aufgabe auf jedem Server durchführen möchten. Standardmäßig arbeitet ShutIt seriell, was sehr lange dauern würde. Wir können jedoch die Aufgaben im Hintergrund laufen lassen um sie zu beschleunigen.

Hier ist ein Beispiel, welches du ausprobieren kannst. Es verwendet den trivialen Befehl: ‘sleep’.

import shutit
import time
long_command="""sleep 60"""
session1 = shutit.create_session('bash')
session2 = shutit.create_session('bash')
password1 = session1.get_input('Password for server1', ispass=True)
password2 = session2.get_input('Password for server2', ispass=True)
session1.login('ssh [email protected]', user='du', password=password1)
session2.login('ssh [email protected]', user='du', password=password2)
start = time.time()
session1.send(long_command, background=True)
session2.send(long_command, background=True)
print('Es hat: ' + str(time.time() - start) + ' Sekunden zum Starten gebraucht')
session1.wait()
session2.wait()
print('Es hat:' + str(time.time() - start) + ' Sekunden zum Vollenden gebraucht')

Mein Computer meint, dass er 0.5 Sekunden gebraucht hat um die Befehle zu starten und dann nur etwas über eine Minute gebraucht um sie zu beenden (mit Verwendung der ‘wait’ Methode).

Das alles ist trivial, aber stelle dir vor das du hunderte an Servern so managen kannst und man kann nun das Potential sehen, die in ein paar Zeilen Code und ein Python import liegen können.

Lerne mehr

Es gibt noch viel mehr, was mit ShutIt erreicht werden kann.

Um mehr zu erfahren, siehe:

ShutIt GitHub

Es handelt sich um ein breiteres Automatiesierungsframework, und das oben genannte ist der sogennante ‘standalone Modus’.

Feedback, feature requests, ‘Wie mache ich es’ sind herzlich willkommen! Erreiche mit unter @ianmiell


Du hast einen Verbesserungsvorschlag oder einen Fehler gefunden? Erstelle ein Ticket im offiziellen GitHub Repo, oder du erstellst einfach gleich einen pull request!

Originalversion von Ian Miell, mit Updates von 2 contributor(s).