---
-- Working Example: https://www.youtube.com/watch?v=PDxTR_KJLhc
-- IMPORTANT: run node.compile("imap.lua") after uploading this script 
-- to create a compiled module. Then run file.remove("imap.lua")
-- @name imap
-- @description An IMAP 4rev1 module that can be used to read email. 
-- Tested on NodeMCU 0.9.5 build 20150213.
-- @date March 12, 2015
-- @author Miguel 
--  GitHub: https://github.com/AllAboutEE 
--  YouTube: https://www.youtube.com/user/AllAboutEE
--  Website: http://AllAboutEE.com
--
-- Visit the following URLs to learn more about IMAP:
-- "How to test an IMAP server by using telnet" http://www.anta.net/misc/telnet-troubleshooting/imap.shtml
-- "RFC 2060 - Internet Message Access Protocol - Version 4rev1" http://www.faqs.org/rfcs/rfc2060.html
-------------------------------------------------------------------------------------------------------------
local moduleName = ... 
local M = {}
_G[moduleName] = M 

local USERNAME = "" 
local PASSWORD = ""

local SERVER = ""
local PORT = ""
local TAG = ""

local DEBUG = false

local body = "" -- used to store an email's body / main text
local header = "" -- used to store an email's last requested header field e.g. SUBJECT, FROM, DATA etc.
local most_recent_num = 1 -- used to store the latest/newest email number/id


local response_processed = false -- used to know if the last IMAP response has been processed

---
-- @name response_processed
-- @returns The response process status of the last IMAP command sent
function M.response_processed()
    return response_processed
end

---
-- @name display
-- @description A generic IMAP response processing function. 
-- Can disply the IMAP response if DEBUG is set to true.
-- Sets the reponse processed variable to true when the string "complete"
-- is found in the IMAP reply/response
local function display(socket, response)

    -- If debuggins is enabled print the IMAP response
    if(DEBUG) then
        print(response)
    end

    -- Some IMAP responses are long enough that they will cause the display 
    -- function to be called several times. One thing is certain, IMAP will replay with
    -- "<tag> OK <command> complete" when it's done sending data back.
    if(string.match(response,'complete') ~= nil) then
        response_processed = true
    end

end

--- 
-- @name config
-- @description Initiates the IMAP settings
function M.config(username,password,tag,debug)
    USERNAME = username
    PASSWORD = password
    TAG = tag
    DEBUG = debug
end

---
-- @name login
-- @descrpiton Logs into a new email session
function M.login(socket)
    response_processed = false -- we are sending a new command 
                               -- which means that the response for it has not been processed
    socket:send(TAG .. " LOGIN " .. USERNAME .. " " .. PASSWORD .. "\r\n")
    socket:on("receive",display)
end

---
-- @name get_most_recent_num
-- @returns The most recent email number. Should only be called after examine()
function M.get_most_recent_num()
    return most_recent_num
end

---
-- @name set_most_recent_num 
-- @description Gets the most recent email number from the EXAMINE command.
-- i.e. if EXAMINE returns "* 4 EXISTS" this means that there are 4 emails,
-- so the latest/newest will be identified by the number 4
local function set_most_recent_num(socket,response)

    if(DEBUG) then
        print(response)
    end

    local _, _, num = string.find(response,"([0-9]+) EXISTS(\.)") -- the _ and _ keep the index of the string found
                                                                  -- but we don't care about that.

    if(num~=nil) then 
        most_recent_num = num
    end

    if(string.match(response,'complete') ~= nil) then
        response_processed = true
    end
end

---
-- @name examine
-- @description IMAP examines the given mailbox/folder. Sends the IMAP EXAMINE command
function M.examine(socket,mailbox)

    response_processed = false
    socket:send(TAG .. " EXAMINE " .. mailbox .. "\r\n")
    socket:on("receive",set_most_recent_num)
end

---
-- @name get_header
-- @returns The last fetched header field
function M.get_header()
    return header
end

---
-- @name set_header
-- @description Records the IMAP header field response in a variable 
-- so that it may be read later
local function set_header(socket,response)
    if(DEBUG) then
        print(response)
    end

    header = header .. response
    if(string.match(response,'complete') ~= nil) then
        response_processed = true
    end
end

---
-- @name fetch_header
-- @description Fetches an emails header field e.g. SUBJECT, FROM, DATE
-- @param socket The IMAP socket to use
-- @param msg_number The email number to read e.g. 1 will read fetch the latest/newest email
-- @param field A header field such as SUBJECT, FROM, or DATE
function M.fetch_header(socket,msg_number,field)
    header = "" -- we are getting a new header so clear this variable
    response_processed = false
    socket:send(TAG .. " FETCH " .. msg_number .. " BODY[HEADER.FIELDS (" .. field .. ")]\r\n")
    socket:on("receive",set_header)
end


---
-- @name get_body
-- @return The last email read's body 
function M.get_body()
    return body
end

---
-- @name set_body
-- @description Records the IMAP body response in a variable 
-- so that it may be read later
local function set_body(socket,response)

    if(DEBUG) then
        print(response)
    end

    body = body .. response
    if(string.match(response,'complete') ~= nil) then
        response_processed = true
    end
end

---
-- @name fetch_body_plain_text
-- @description Sends the IMAP command to fetch a plain text version of the email's body
-- @param socket The IMAP socket to use
-- @param msg_number The email number to obtain e.g. 1 will obtain the latest email
function M.fetch_body_plain_text(socket,msg_number)
    response_processed = false
    body = "" -- clear the body variable since we'll be fetching a  new email
    socket:send(TAG .. " FETCH " .. msg_number .. " BODY[1]\r\n")
    socket:on("receive",set_body)
end

---
-- @name logout
-- @description Sends the IMAP command to logout of the email session
function M.logout(socket)
    response_processed = false
    socket:send(TAG .. " LOGOUT\r\n")
    socket:on("receive",display)
end