Advertisement
hbar

cleverbot

Jun 6th, 2013
4,922
0
Never
Not a member of Pastebin yet? Sign Up, it unlocks many cool features!
Lua 15.02 KB | None | 0 0
  1. -- ATTENTION: THIS LIBRARY DOESN'T CURRENTLY WORK DUE TO CHANCES IN THE CLEVERBOT API
  2.  
  3. --[[
  4. Cleverbot client for ComputerCraft v1.4
  5. Based on https://github.com/folz/cleverbot.py
  6.  
  7. This code uses a pure lua implementation of the MD5 hash algorithm by Adam Baldwin as presented
  8. in this Stack Overflow answer: http://stackoverflow.com/a/12292659
  9.  
  10. Cleverbot code begins around line 420.
  11.  
  12. Usage:
  13.     pastebin get Dv9x1ppc cleverbot
  14.     cleverbot
  15.  
  16.  
  17. N.B. The link below is for an old version, currently there's no chatbox integration!
  18.  
  19. See the code in action: http://imgur.com/a/MNN4J
  20.  
  21. This same file can also be used as an API.
  22.  
  23. Example code:
  24.         os.loadAPI("cleverbot")
  25.  
  26.         bot = cleverbot.Cleverbot.new()
  27.  
  28.         message = "Hi there!"
  29.         response = bot:send(message)
  30.         print(response)
  31.  
  32. Notes:
  33.     * Sending the message is pretty slow as the hashing takes time and cleverbot.com can sometimes take a while to respond.
  34.     On slower computers this can sometimes lead to "too long without yielding" error.
  35.  
  36. Matti Vapa (2015)
  37. ]]--
  38.  
  39.  
  40. -- Here are the (slightly modified) LuaBit and MD5 implementations with original comments
  41.  
  42. --[[---------------
  43. LuaBit v0.4
  44. -------------------
  45. a bitwise operation lib for lua.
  46.  
  47. http://luaforge.net/projects/bit/
  48.  
  49. How to use:
  50. -------------------
  51.  bit.bnot(n) -- bitwise not (~n)
  52.  bit.band(m, n) -- bitwise and (m & n)
  53.  bit.bor(m, n) -- bitwise or (m | n)
  54.  bit.bxor(m, n) -- bitwise xor (m ^ n)
  55.  bit.brshift(n, bits) -- right shift (n >> bits)
  56.  bit.blshift(n, bits) -- left shift (n << bits)
  57.  bit.blogic_rshift(n, bits) -- logic right shift(zero fill >>>)
  58.  
  59. Please note that bit.brshift and bit.blshift only support number within
  60. 32 bits.
  61.  
  62. 2 utility functions are provided too:
  63.  bit.tobits(n) -- convert n into a bit table(which is a 1/0 sequence)
  64.                -- high bits first
  65.  bit.tonumb(bit_tbl) -- convert a bit table into a number
  66. -------------------
  67.  
  68. Under the MIT license.
  69.  
  70. copyright(c) 2006~2007 hanzhao ([email protected])
  71. --]]---------------
  72.  
  73. --do
  74.  
  75. ------------------------
  76. -- bit lib implementions
  77.  
  78. local function check_int(n)
  79.  -- checking not float
  80.  if(n - math.floor(n) > 0) then
  81.   error("trying to use bitwise operation on non-integer!")
  82.  end
  83. end
  84.  
  85. local function tbl_to_number(tbl)
  86.  local n = #tbl
  87.  
  88.  local rslt = 0
  89.  local power = 1
  90.  for i = 1, n do
  91.   rslt = rslt + tbl[i]*power
  92.   power = power*2
  93.  end
  94.  
  95.  return rslt
  96. end
  97.  
  98. local function expand(tbl_m, tbl_n)
  99.  local big = {}
  100.  local small = {}
  101.  if(#tbl_m > #tbl_n) then
  102.   big = tbl_m
  103.   small = tbl_n
  104.  else
  105.   big = tbl_n
  106.   small = tbl_m
  107.  end
  108.  -- expand small
  109.  for i = #small + 1, #big do
  110.   small[i] = 0
  111.  end
  112.  
  113. end
  114.  
  115. local to_bits = function () end
  116.  
  117. local function bit_not(n)
  118.  local tbl = to_bits(n)
  119.  local size = math.max(#tbl, 32)
  120.  for i = 1, size do
  121.   if(tbl[i] == 1) then
  122.    tbl[i] = 0
  123.   else
  124.    tbl[i] = 1
  125.   end
  126.  end
  127.  return tbl_to_number(tbl)
  128. end
  129.  
  130.  
  131. to_bits = function (n)
  132.  check_int(n)
  133.  if(n < 0) then
  134.   -- negative
  135.   return to_bits(bit_not(math.abs(n)) + 1)
  136.  end
  137.  -- to bits table
  138.  local tbl = {}
  139.  local cnt = 1
  140.  while (n > 0) do
  141.   local last = math.fmod(n,2)
  142.   if(last == 1) then
  143.    tbl[cnt] = 1
  144.   else
  145.    tbl[cnt] = 0
  146.   end
  147.   n = (n-last)/2
  148.   cnt = cnt + 1
  149.  end
  150.  
  151.  return tbl
  152. end
  153.  
  154.  
  155. local function bit_or(m, n)
  156.  local tbl_m = to_bits(m)
  157.  local tbl_n = to_bits(n)
  158.  expand(tbl_m, tbl_n)
  159.  
  160.  local tbl = {}
  161.  local rslt = math.max(#tbl_m, #tbl_n)
  162.  for i = 1, rslt do
  163.   if(tbl_m[i]== 0 and tbl_n[i] == 0) then
  164.    tbl[i] = 0
  165.   else
  166.    tbl[i] = 1
  167.   end
  168.  end
  169.  
  170.  return tbl_to_number(tbl)
  171. end
  172.  
  173. local function bit_and(m, n)
  174.  local tbl_m = to_bits(m)
  175.  local tbl_n = to_bits(n)
  176.  expand(tbl_m, tbl_n)
  177.  
  178.  local tbl = {}
  179.  local rslt = math.max(#tbl_m, #tbl_n)
  180.  for i = 1, rslt do
  181.   if(tbl_m[i]== 0 or tbl_n[i] == 0) then
  182.    tbl[i] = 0
  183.   else
  184.    tbl[i] = 1
  185.   end
  186.  end
  187.  
  188.  return tbl_to_number(tbl)
  189. end
  190.  
  191. local function bit_xor(m, n)
  192.  local tbl_m = to_bits(m)
  193.  local tbl_n = to_bits(n)
  194.  expand(tbl_m, tbl_n)
  195.  
  196.  local tbl = {}
  197.  local rslt = math.max(#tbl_m, #tbl_n)
  198.  for i = 1, rslt do
  199.   if(tbl_m[i] ~= tbl_n[i]) then
  200.    tbl[i] = 1
  201.   else
  202.    tbl[i] = 0
  203.   end
  204.  end
  205.  
  206.  --table.foreach(tbl, print)
  207.  
  208.  return tbl_to_number(tbl)
  209. end
  210.  
  211. local function bit_rshift(n, bits)
  212.  check_int(n)
  213.  
  214.  local high_bit = 0
  215.  if(n < 0) then
  216.   -- negative
  217.   n = bit_not(math.abs(n)) + 1
  218.   high_bit = 2147483648 -- 0x80000000
  219.  end
  220.  
  221.  for i=1, bits do
  222.   n = n/2
  223.   n = bit_or(math.floor(n), high_bit)
  224.  end
  225.  return math.floor(n)
  226. end
  227.  
  228. -- logic rightshift assures zero filling shift
  229. local function bit_logic_rshift(n, bits)
  230.  check_int(n)
  231.  if(n < 0) then
  232.   -- negative
  233.   n = bit_not(math.abs(n)) + 1
  234.  end
  235.  for i=1, bits do
  236.   n = n/2
  237.  end
  238.  return math.floor(n)
  239. end
  240.  
  241. local function bit_lshift(n, bits)
  242.  check_int(n)
  243.  
  244.  if(n < 0) then
  245.   -- negative
  246.   n = bit_not(math.abs(n)) + 1
  247.  end
  248.  
  249.  for i=1, bits do
  250.   n = n*2
  251.  end
  252.  return bit_and(n, 4294967295) -- 0xFFFFFFFF
  253. end
  254.  
  255. local function bit_xor2(m, n)
  256.  local rhs = bit_or(bit_not(m), bit_not(n))
  257.  local lhs = bit_or(m, n)
  258.  local rslt = bit_and(lhs, rhs)
  259.  return rslt
  260. end
  261.  
  262. -- An MD5 mplementation in Lua, requires bitlib (hacked to use LuaBit from above, ugh)
  263. -- 10/02/2001 [email protected]
  264.  
  265. local md5={ff=tonumber('ffffffff',16),consts={}}
  266.  
  267. string.gsub([[ d76aa478 e8c7b756 242070db c1bdceee
  268.     f57c0faf 4787c62a a8304613 fd469501
  269.     698098d8 8b44f7af ffff5bb1 895cd7be
  270.     6b901122 fd987193 a679438e 49b40821
  271.     f61e2562 c040b340 265e5a51 e9b6c7aa
  272.     d62f105d 02441453 d8a1e681 e7d3fbc8
  273.     21e1cde6 c33707d6 f4d50d87 455a14ed
  274.     a9e3e905 fcefa3f8 676f02d9 8d2a4c8a
  275.     fffa3942 8771f681 6d9d6122 fde5380c
  276.     a4beea44 4bdecfa9 f6bb4b60 bebfbc70
  277.     289b7ec6 eaa127fa d4ef3085 04881d05
  278.     d9d4d039 e6db99e5 1fa27cf8 c4ac5665
  279.     f4292244 432aff97 ab9423a7 fc93a039
  280.     655b59c3 8f0ccc92 ffeff47d 85845dd1
  281.     6fa87e4f fe2ce6e0 a3014314 4e0811a1
  282.     f7537e82 bd3af235 2ad7d2bb eb86d391
  283.     67452301 efcdab89 98badcfe 10325476 ]],"(%w+)", function (s) table.insert(md5.consts, tonumber(s,16)) end)
  284.     --67452301 efcdab89 98badcfe 10325476 ]],"(%w+)", function (s) tinsert(md5.consts,tonumber(s,16)) end)
  285.  
  286. function transform(A,B,C,D,X)
  287.   local f=function (x,y,z) return bit_or(bit_and(x,y),bit_and(-x-1,z)) end
  288.   local g=function (x,y,z) return bit_or(bit_and(x,z),bit_and(y,-z-1)) end
  289.   local h=function (x,y,z) return bit_xor(x,bit_xor(y,z)) end
  290.   local i=function (x,y,z) return bit_xor(y,bit_or(x,-z-1)) end
  291.   local z=function (f,a,b,c,d,x,s,ac)
  292.         a=bit_and(a+f(b,c,d)+x+ac,md5.ff)
  293.         -- be *very* careful that left shift does not cause rounding!
  294.         return bit_or(bit_lshift(bit_and(a,bit_rshift(md5.ff,s)),s),bit_rshift(a,32-s))+b
  295.       end
  296.   local a,b,c,d=A,B,C,D
  297.   local t=md5.consts
  298.  
  299.   a=z(f,a,b,c,d,X[ 0], 7,t[ 1])
  300.   d=z(f,d,a,b,c,X[ 1],12,t[ 2])
  301.   c=z(f,c,d,a,b,X[ 2],17,t[ 3])
  302.   b=z(f,b,c,d,a,X[ 3],22,t[ 4])
  303.   a=z(f,a,b,c,d,X[ 4], 7,t[ 5])
  304.   d=z(f,d,a,b,c,X[ 5],12,t[ 6])
  305.   c=z(f,c,d,a,b,X[ 6],17,t[ 7])
  306.   b=z(f,b,c,d,a,X[ 7],22,t[ 8])
  307.   a=z(f,a,b,c,d,X[ 8], 7,t[ 9])
  308.   d=z(f,d,a,b,c,X[ 9],12,t[10])
  309.   c=z(f,c,d,a,b,X[10],17,t[11])
  310.   b=z(f,b,c,d,a,X[11],22,t[12])
  311.   a=z(f,a,b,c,d,X[12], 7,t[13])
  312.   d=z(f,d,a,b,c,X[13],12,t[14])
  313.   c=z(f,c,d,a,b,X[14],17,t[15])
  314.   b=z(f,b,c,d,a,X[15],22,t[16])
  315.  
  316.   a=z(g,a,b,c,d,X[ 1], 5,t[17])
  317.   d=z(g,d,a,b,c,X[ 6], 9,t[18])
  318.   c=z(g,c,d,a,b,X[11],14,t[19])
  319.   b=z(g,b,c,d,a,X[ 0],20,t[20])
  320.   a=z(g,a,b,c,d,X[ 5], 5,t[21])
  321.   d=z(g,d,a,b,c,X[10], 9,t[22])
  322.   c=z(g,c,d,a,b,X[15],14,t[23])
  323.   b=z(g,b,c,d,a,X[ 4],20,t[24])
  324.   a=z(g,a,b,c,d,X[ 9], 5,t[25])
  325.   d=z(g,d,a,b,c,X[14], 9,t[26])
  326.   c=z(g,c,d,a,b,X[ 3],14,t[27])
  327.   b=z(g,b,c,d,a,X[ 8],20,t[28])
  328.   a=z(g,a,b,c,d,X[13], 5,t[29])
  329.   d=z(g,d,a,b,c,X[ 2], 9,t[30])
  330.   c=z(g,c,d,a,b,X[ 7],14,t[31])
  331.   b=z(g,b,c,d,a,X[12],20,t[32])
  332.  
  333.   a=z(h,a,b,c,d,X[ 5], 4,t[33])
  334.   d=z(h,d,a,b,c,X[ 8],11,t[34])
  335.   c=z(h,c,d,a,b,X[11],16,t[35])
  336.   b=z(h,b,c,d,a,X[14],23,t[36])
  337.   a=z(h,a,b,c,d,X[ 1], 4,t[37])
  338.   d=z(h,d,a,b,c,X[ 4],11,t[38])
  339.   c=z(h,c,d,a,b,X[ 7],16,t[39])
  340.   b=z(h,b,c,d,a,X[10],23,t[40])
  341.   a=z(h,a,b,c,d,X[13], 4,t[41])
  342.   d=z(h,d,a,b,c,X[ 0],11,t[42])
  343.   c=z(h,c,d,a,b,X[ 3],16,t[43])
  344.   b=z(h,b,c,d,a,X[ 6],23,t[44])
  345.   a=z(h,a,b,c,d,X[ 9], 4,t[45])
  346.   d=z(h,d,a,b,c,X[12],11,t[46])
  347.   c=z(h,c,d,a,b,X[15],16,t[47])
  348.   b=z(h,b,c,d,a,X[ 2],23,t[48])
  349.  
  350.   a=z(i,a,b,c,d,X[ 0], 6,t[49])
  351.   d=z(i,d,a,b,c,X[ 7],10,t[50])
  352.   c=z(i,c,d,a,b,X[14],15,t[51])
  353.   b=z(i,b,c,d,a,X[ 5],21,t[52])
  354.   a=z(i,a,b,c,d,X[12], 6,t[53])
  355.   d=z(i,d,a,b,c,X[ 3],10,t[54])
  356.   c=z(i,c,d,a,b,X[10],15,t[55])
  357.   b=z(i,b,c,d,a,X[ 1],21,t[56])
  358.   a=z(i,a,b,c,d,X[ 8], 6,t[57])
  359.   d=z(i,d,a,b,c,X[15],10,t[58])
  360.   c=z(i,c,d,a,b,X[ 6],15,t[59])
  361.   b=z(i,b,c,d,a,X[13],21,t[60])
  362.   a=z(i,a,b,c,d,X[ 4], 6,t[61])
  363.   d=z(i,d,a,b,c,X[11],10,t[62])
  364.   c=z(i,c,d,a,b,X[ 2],15,t[63])
  365.   b=z(i,b,c,d,a,X[ 9],21,t[64])
  366.  
  367.   return A+a,B+b,C+c,D+d
  368. end
  369.  
  370. -- convert little-endian 32-bit int to a 4-char string
  371. local function leIstr(i)
  372.   local f=function (s) return string.char(bit_and(bit_rshift(i,s),255)) end
  373.   return f(0)..f(8)..f(16)..f(24)
  374. end
  375.  
  376.   -- convert raw string to big-endian int
  377.   local function beInt(s)
  378.     local v=0
  379.     for i=1,string.len(s) do v=v*256+string.byte(s,i) end
  380.     return v
  381.   end
  382.   -- convert raw string to little-endian int
  383.   local function leInt(s)
  384.     local v=0
  385.     for i=string.len(s),1,-1 do v=v*256+string.byte(s,i) end
  386.     return v
  387.   end
  388.   -- cut up a string in little-endian ints of given size
  389.   local function leStrCuts(s,...)
  390.     local o,r=1,{}
  391.     for i=1,#arg do
  392.       table.insert(r,leInt(string.sub(s,o,o+arg[i]-1)))
  393.       o=o+arg[i]
  394.     end
  395.     return r
  396.   end
  397.  
  398. function Calc(s)
  399.   local msgLen=string.len(s)
  400.   local padLen=56- msgLen % 64
  401.   if msgLen % 64 > 56 then padLen=padLen+64 end
  402.   if padLen==0 then padLen=64 end
  403.   s=s..string.char(128)..string.rep(string.char(0),padLen-1)
  404.   s=s..leIstr(8*msgLen)..leIstr(0)
  405.   assert(string.len(s) % 64 ==0)
  406.   local t=md5.consts
  407.   local a,b,c,d=t[65],t[66],t[67],t[68]
  408.   for i=1,string.len(s),64 do
  409.     local X=leStrCuts(string.sub(s,i,i+63),4,4,4,4,4,4,4,4,4,4,4,4,4,4,4,4)
  410.     assert(#X==16)
  411.     X[0]=table.remove(X,1) -- zero based!
  412.     a,b,c,d=transform(a,b,c,d,X)
  413.   end
  414.   local swap=function (w) return beInt(leIstr(w)) end
  415.   return string.format("%08x%08x%08x%08x",swap(a),swap(b),swap(c),swap(d))
  416. end
  417.  
  418.  
  419.  
  420.  
  421.  
  422.  
  423.  
  424. -- Here begins the Cleverbot client section.
  425.  
  426.  
  427. -- Utility functions
  428.  
  429. -- parse the response lines to a table
  430. split2lines = function(input)
  431.   local output = {}
  432.   local i = 0
  433.   local line = input.readLine()
  434.   while line ~= nil do
  435.     output[i] = line
  436.     i = i + 1
  437.     line = input.readLine()
  438.   end
  439.   return output
  440. end
  441.  
  442. -- Encode the variables to a string so that
  443. -- vars["key"] = "value" becomes "key=value&"
  444. encodeVars = function(vars)
  445.   local s = ""
  446.   for k,v in pairs(vars) do
  447.     if #v > 0 then
  448.       s = s..k.."="..textutils.urlEncode(v).."&"
  449.     end
  450.   end
  451.   -- remove trailing '&' before returning
  452.   return s:sub(1,#s-1)
  453. end
  454.  
  455.  
  456. URL = "http://www.cleverbot.com/webservicemin"
  457.  
  458. -- headers
  459.  
  460. HEADERS = {}
  461. HEADERS['User-Agent'] = 'Mozilla/4.0 (compatible; MSIE 8.0; Windows NT 6.0)'
  462. HEADERS['Accept'] = 'text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8'
  463. HEADERS['Accept-Charset'] = 'ISO-8859-1,utf-8;q=0.7,*;q=0.7'
  464. HEADERS['Accept-Language'] = 'en-us,en;q=0.8,en-us;q=0.5,en;q=0.3'
  465. HEADERS['Cache-Control'] = 'no-cache'
  466. HEADERS['Host'] = 'www.cleverbot.com'
  467. HEADERS['Referer'] = 'http://www.cleverbot.com/'
  468. HEADERS['Pragma'] = 'no-cache'
  469. HEADERS['Cookie'] = 'XVIS=TEI939AFFIAGAYQZ'
  470.  
  471.  
  472.  
  473. -- initial variables
  474. vars = { }
  475. vars['stimulus'] = ''
  476. vars['start'] = 'y'
  477. vars['sessionid'] = ''
  478. vars['vText8'] = ''
  479. vars['vText7'] = ''
  480. vars['vText6'] = ''
  481. vars['vText5'] = ''
  482. vars['vText4'] = ''
  483. vars['vText3'] = ''
  484. vars['vText2'] = ''
  485. vars['icognoid'] = 'wsf'
  486. vars['icognocheck'] = ''
  487. vars['fno'] = "0"
  488. vars['prevref'] = ''
  489. vars['emotionaloutput'] = ''
  490. vars['emotionalhistory'] = ''
  491. vars['asbotname'] = ''
  492. vars['ttsvoice'] = ''
  493. vars['typing'] = ''
  494. vars['lineref'] = ''
  495. vars['sub'] = 'Say'
  496. vars['islearning'] = "1"
  497. vars['cleanslate'] = "False"
  498.  
  499.  
  500. -- Here is the Cleverbot "class" that can be used by an external program
  501. -- or by the client at the end of the file.
  502.  
  503. Cleverbot = {}
  504.  
  505. Cleverbot.new = function()
  506.   local bot = {}
  507.   local tvars = {}
  508.   for k,v in pairs(vars) do
  509.     tvars[k] = v
  510.   end
  511.   bot.vars = tvars
  512.   bot.conversation = {}
  513.   bot.response = ""
  514.   setmetatable(bot,{__index = Cleverbot})
  515.   return bot
  516. end
  517.  
  518. -- This method is used for sending a message to the Cleverbot backend.
  519. -- The bot keeps track of previous conversation automatically.
  520.  
  521. Cleverbot.send = function(self,message)
  522.   self.vars["stimulus"] = message
  523.   table.insert(self.conversation, message)
  524.   local response = self:_send()
  525.   local parsed
  526.   if response ~= "error" then
  527.     self.response = response
  528.     parsed = self:_parse()
  529.   else
  530.     return "Error communicating with Cleverbot!"
  531.   end
  532.  
  533.   if self.vars['sessionid'] ~= "" then
  534.     self.vars['sessionid'] = parsed['sessionid']
  535.   end
  536.   table.insert(self.conversation,parsed['answer'])
  537.  
  538.   return parsed['answer']
  539.  
  540. end
  541.  
  542. -- This is the internal method that handles the actual communication
  543. -- with the backend. You shouldn't need to use this yourself.
  544.  
  545. Cleverbot._send = function(self)
  546.  
  547.   local n = #self.conversation
  548.   local i = 1
  549.  
  550.   for j=n,1,-1 do
  551.     i = i + 1
  552.     self.vars['vText'..tostring(i)] = self.conversation[j]
  553.     if i == 8 then
  554.       break
  555.     end
  556.   end
  557.  
  558.   -- I'll quote a cleverbot.py comment the explain why the data is encoded twice
  559.  
  560.   --[[
  561.       Cleverbot tries to prevent unauthorized access to its API by
  562.       obfuscating how it generates the 'icognocheck' token, so we have
  563.       to URLencode the data twice: once to generate the token, and
  564.       twice to add the token to the data we're sending to Cleverbot.
  565.   ]]--
  566.  
  567.   local enc_data = encodeVars(self.vars)
  568.   local digest = enc_data:sub(10,35)
  569.   local hash = Calc(digest)
  570.   self.vars['icognocheck'] = hash
  571.   enc_data = encodeVars(self.vars)
  572.  
  573.   local response = http.post(URL,enc_data,HEADERS)
  574.  
  575.   if not response then return "error" end
  576.  
  577.   if response.getResponseCode() ~= 200 then
  578.     return "error"
  579.   else
  580.     return response
  581.   end
  582. end
  583.  
  584. -- This is the internal method that is used to parse the response
  585. -- and extract all the relevant data.
  586.  
  587. Cleverbot._parse = function(self)
  588.   local lines = split2lines(self.response)
  589.   local parsed = {}
  590.   parsed['conversation_id'] = lines[1]
  591.   parsed['conversation_log_id'] = lines[2]
  592.   parsed['answer'] = lines[0]
  593.   return parsed
  594. end
  595.  
  596.  
  597. -- The rest of the file is a simple standalone client. You can also use just the Cleverbot
  598. -- class in you own code. Check the beginning of this file for instructions.
  599.  
  600.  
  601.  
  602.  
  603. -- If the script was run in the shell, we start the standalone client.
  604. if shell then
  605.  
  606.   local bot = Cleverbot.new()
  607.  
  608.   -- main loop
  609.   local m = ""
  610.   while true do
  611.     term.write("You: ")
  612.     m = read()
  613.     m = bot:send(m)
  614.     print("Cleverbot: "..m)
  615.     sleep(0.1)
  616.   end
  617.  
  618. end
Advertisement
Add Comment
Please, Sign In to add comment
Advertisement