aboutsummaryrefslogtreecommitdiff
path: root/4.2/get_mail.rb
diff options
context:
space:
mode:
authorNat Lasseter <user@4574.co.uk>2025-02-19 23:14:40 +0000
committerNat Lasseter <user@4574.co.uk>2025-02-19 23:14:40 +0000
commitc296a63e974e04c3c31f531747594d9e82723f45 (patch)
tree6d6f82ba3bb58abeb4f563eb0be0983daaa8c86b /4.2/get_mail.rb
parente532f817c53589de47ead76618677db1ddbb2e48 (diff)
[4.2] Authenticators have lifespans now. All done!
Diffstat (limited to '4.2/get_mail.rb')
-rwxr-xr-x4.2/get_mail.rb118
1 files changed, 118 insertions, 0 deletions
diff --git a/4.2/get_mail.rb b/4.2/get_mail.rb
new file mode 100755
index 0000000..88167bf
--- /dev/null
+++ b/4.2/get_mail.rb
@@ -0,0 +1,118 @@
+#!/usr/bin/env ruby
+
+require 'net/http'
+require 'uri'
+require 'json'
+
+def encrypt(obj, key)
+ cipher = OpenSSL::Cipher::AES.new(256, :CBC).encrypt
+ cipher.key = Digest::SHA2.digest(key)
+ s = cipher.update(obj) + cipher.final
+ s.unpack('H*')[0].upcase
+end
+
+def decrypt(obj, key)
+ ticket = [obj].pack("H*").unpack("C*").pack("c*")
+ cipher = OpenSSL::Cipher::AES.new(256, :CBC).decrypt
+ cipher.key = Digest::SHA2.digest(key)
+ cipher.update(ticket) + cipher.final
+end
+
+def gen_auth(un, ws, sk)
+ ts = Time.now.to_i
+ encrypt("#{un},#{ws},120,#{ts}", sk)
+end
+
+def get_ticket(un, wsa, tgt, sv, sk)
+ uri = URI.parse("http://localhost:4567/ticket")
+ header = {'Content-Type': 'text/json'}
+
+ login = {
+ "username": un,
+ "ticket": tgt,
+ "authenticator": gen_auth(un, wsa, sk),
+ "service": sv
+ }
+
+ http = Net::HTTP.new(uri.host, uri.port)
+ request = Net::HTTP::Post.new(uri.request_uri, header)
+ request.body = login.to_json
+
+ response = http.request(request)
+ response.body
+end
+
+def ticket_valid?(ticket)
+ now = Time.now.to_i
+ timeStart = ticket["timestamp"]
+ timeEnd = ticket["timestamp"] + ticket["lifespan"]
+ now >= timeStart && now < timeEnd
+end
+
+def update_keytab!
+ File.open(".keytab", "w") do |f|
+ f.puts Tickets.map { |s, p| [
+ s,
+ p["sessionkey"],
+ p["ws_address"],
+ p["lifespan"],
+ p["timestamp"],
+ p["ticket"]
+ ].join(?,) }
+ end
+end
+
+print "Username: "; un = gets.strip
+print "Mailserver: "; ms = gets.strip
+
+unless File.exist?(".keytab")
+ puts "No keytab, please kinit"
+ exit 1
+end
+
+Tickets = File.readlines(".keytab").map { |l|
+ a = l.strip.split(?,)
+ [a[0], {
+ "sessionkey" => a[1],
+ "ws_address" => a[2],
+ "lifespan" => a[3].to_i,
+ "timestamp" => a[4].to_i,
+ "ticket" => a[5]
+ }]
+}.to_h
+
+unless Tickets.keys.include?(ms) && ticket_valid?(Tickets[ms])
+ if Tickets.keys.include?("_TGS") && ticket_valid?(Tickets["_TGS"])
+ packet = get_ticket(un, Tickets["_TGS"]["ws_address"],
+ Tickets["_TGS"]["ticket"],
+ ms, Tickets["_TGS"]["sessionkey"])
+ packet = decrypt(packet, Tickets["_TGS"]["sessionkey"]).split(?,)
+ Tickets[ms] = {
+ "sessionkey" => packet[0],
+ "ws_address" => packet[1],
+ "lifespan" => packet[2].to_i,
+ "timestamp" => packet[3].to_i,
+ "ticket" => packet[4]
+ }
+ update_keytab!
+ else
+ puts "No Ticket Granting Ticket, please kinit"
+ exit 1
+ end
+end
+
+uri = URI.parse("http://localhost:4568/login")
+header = {'Content-Type': 'text/json'}
+
+login = {
+ "username": un,
+ "ticket": Tickets[ms]["ticket"],
+ "authenticator": gen_auth(un, Tickets[ms]["ws_address"], Tickets[ms]["sessionkey"])
+}
+
+http = Net::HTTP.new(uri.host, uri.port)
+request = Net::HTTP::Post.new(uri.request_uri, header)
+request.body = login.to_json
+
+response = http.request(request)
+puts response.body