File: auth_tkt.rb

package info (click to toggle)
libapache2-mod-auth-tkt 2.3.99~b1-1
  • links: PTS
  • area: main
  • in suites: bookworm, bullseye, buster, forky, sid, trixie
  • size: 856 kB
  • sloc: ansic: 2,213; perl: 1,756; sh: 128; makefile: 94; php: 88; ruby: 59; tcl: 55; python: 48
file content (118 lines) | stat: -rw-r--r-- 3,705 bytes parent folder | download
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
########################################################################
#
# File: auth_tkt.rb
# By:   Sascha Hanssen, MESO Web Scapes (hanssen@meso.net, www.meso.net)
# Date: 2008-01-23
#
########################################################################
#
########################################################################
#
# This file defines functions to generate cookie tickets compatible
# with the "mod_auth_tkt" apache module.
#
# Save this file to your RailsApplication/lib folder
# Include functionallity with "include AuthTkt" into your controller
#
########################################################################

module AuthTkt
  # set path to auth_tkt config file, where TKTAuthSecret is set
  SECRET_KEY_FILE = "/path/to/file.conf";

  # set root domain to be able to single sign on (SSO)
  # (access all subdomains with one valid ticket)
  DOMAIN = ".yourdomain.com"

  # sets the auth_tkt cookie, returns the signed cookie string
  def set_auth_tkt_cookie(user, domain = nil, token_list = nil, user_data = nil, base64 = false)
    # get signed cookie string
    tkt_hash = get_tkt_hash(user, token_list, user_data, base64)

    cookie_data = { :value => tkt_hash }

    # set domain for cookie, if wanted
    cookie_data[:domain] = domain if domain

    # store data into cookie
    cookies[:auth_tkt] = cookie_data

    # return signed cookie
    return tkt_hash
  end

  # destroys the auth_tkt, to log an user out
  def destroy_auth_tkt_cookie
    # reset ticket value of cookie, to log out even if deleting cookie fails
    cookies[:auth_tkt] = { :value => '', :expire => Time.at(0), :domain => DOMAIN }
    cookies.delete :auth_tkt
  end

  # returns a string that contains the signed cookie content
  # data encryption is not implemented yet, ssl communication is
  # highly recommended, when tokens or user data will be used
  def get_tkt_hash(user, token_list = nil, user_data = nil, base64 = false)
    # ensure payload is not nil
    token_list ||= ''
    user_data  ||= ''

    # set timestamp and binary string for timestamp and ip packed together
    timestamp = Time.now.to_i
    ip_timestamp = [ip2long(request.remote_ip), timestamp].pack("NN")

    # creating the cookie signature
    digest0 = Digest::MD5.hexdigest(ip_timestamp + get_secret_key + user +
                                    "\0" + token_list + "\0" + user_data)

    digest = Digest::MD5.hexdigest(digest0 + get_secret_key)

    # concatenating signature, timestamp and payload
    cookie = digest + timestamp.to_hex + user
    if token_list
      cookie += '!' + token_list
    end
    cookie += '!' + user_data

    # base64 encode cookie, if needed
    if base64
      require 'base64'
      cookie = Base64.b64encode(cookie).gsub("\n", '').strip
    end

    return cookie
  end

  # returns the shared secret string used to sign the cookie
  # read from the scret key file, returns empty string on errors
  def get_secret_key
    secret_key = ''
    return '' unless File.file? SECRET_KEY_FILE
    open(SECRET_KEY_FILE) do |file|
      file.each do |line|
        if line.include? 'TKTAuthSecret'
          secret_key = line.gsub('TKTAuthSecret', '').strip.gsub("\"", '')
          break
        end
      end
    end
    secret_key
  end

  # function adapted according to php: generates an IPv4 Internet network address
  # from its Internet standard format (dotted string) representation.
  def ip2long(ip)
    long = 0
    ip.split( /\./ ).reverse.each_with_index do |x, i|
      long += x.to_i << ( i * 8 )
    end
    long
  end
end

# this class definition may be moved to application.rb
# but works fine here, too
class Integer
  def to_hex
    self.to_s(16)
  end
end