use std::ffi::OsStr;
use std::fs::File;
use std::io::{Stdout, Write};
use std::path::{Path, PathBuf};
use std::process;
use std::process::Stdio;
use crate::config::ErgConfig;
use crate::consts::{EXPERIMENTAL_MODE, PYTHON_MODE};
use crate::env::{
    erg_path, erg_pkgs_path, erg_pystd_path, erg_std_path, python_site_packages, python_sys_path,
};
use crate::pathutil::{add_postfix_foreach, remove_postfix};
use crate::random::random;
use crate::stdin::GLOBAL_STDIN;
use crate::vfs::VFS;
use crate::{normalize_path, power_assert};
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct DummyStdin {
    pub name: String,
    current_line: usize,
    lines: Vec<String>,
}
impl DummyStdin {
    pub fn new(name: String, lines: Vec<String>) -> Self {
        Self {
            name,
            current_line: 0,
            lines,
        }
    }
    pub fn read_line(&mut self) -> String {
        let mut stdout = std::io::stdout();
        if self.current_line >= self.lines.len() {
            stdout.write_all("\n".as_bytes()).unwrap();
            stdout.flush().unwrap();
                        return "exit()".to_string();
        }
        let mut line = self.lines[self.current_line].clone();
        self.current_line += 1;
        line.push('\n');
        stdout.write_all(line.as_bytes()).unwrap();
        stdout.flush().unwrap();
        line
    }
    pub fn reread_lines(&self, ln_begin: usize, ln_end: usize) -> Vec<String> {
        self.lines[ln_begin - 1..=ln_end - 1].to_vec()
    }
    pub fn reread(&self) -> Option<String> {
        self.lines.get(self.current_line).cloned()
    }
}
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub enum InputKind {
    File(PathBuf),
    REPL,
    DummyREPL(DummyStdin),
        Pipe(String),
        Str(String),
    Dummy,
}
impl InputKind {
    pub const fn is_repl(&self) -> bool {
        matches!(self, Self::REPL | Self::DummyREPL(_))
    }
    pub fn path(&self) -> &Path {
        match self {
            Self::File(filename) => filename.as_path(),
            Self::REPL | Self::Pipe(_) => Path::new("<stdin>"),
            Self::DummyREPL(_stdin) => Path::new("<stdin>"),
            Self::Str(_) => Path::new("<string>"),
            Self::Dummy => Path::new("<dummy>"),
        }
    }
    pub fn as_str(&self) -> &str {
        match self {
            Self::File(filename) => filename.to_str().unwrap_or("_"),
            Self::REPL | Self::DummyREPL(_) | Self::Pipe(_) => "<stdin>",
            Self::Str(_) => "<string>",
            Self::Dummy => "<dummy>",
        }
    }
    pub fn dir(&self) -> PathBuf {
        if let Self::File(path) = self {
            let mut path = path.clone();
            path.pop();
            if path.ends_with("__pycache__") {
                path.pop();
            }
            if path.parent().is_none() {
                PathBuf::from(".")
            } else {
                path
            }
        } else {
            PathBuf::from(".")
        }
    }
    pub fn project_root(&self) -> Option<PathBuf> {
        if let Self::File(path) = self {
            let mut parent = path.clone();
            while parent.pop() {
                if parent.join("package.er").exists() {
                    return Some(parent);
                }
            }
            None
        } else {
            None
        }
    }
}
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct Input {
    pub(crate) kind: InputKind,
        id: u64,
}
impl From<PathBuf> for Input {
    fn from(path: PathBuf) -> Self {
        Self::file(path)
    }
}
impl From<&Path> for Input {
    fn from(path: &Path) -> Self {
        Self::file(path.to_path_buf())
    }
}
impl Input {
    pub const fn new(kind: InputKind, id: u64) -> Self {
        Self { kind, id }
    }
    pub fn file(path: PathBuf) -> Self {
        Self::new(InputKind::File(path), random())
    }
    pub fn pipe(src: String) -> Self {
        Self::new(InputKind::Pipe(src), random())
    }
    pub fn str(src: String) -> Self {
        Self::new(InputKind::Str(src), random())
    }
    pub fn repl() -> Self {
        Self::new(InputKind::REPL, random())
    }
    pub fn dummy() -> Self {
        Self::new(InputKind::Dummy, random())
    }
    pub fn dummy_repl(stdin: DummyStdin) -> Self {
        Self::new(InputKind::DummyREPL(stdin), random())
    }
    pub const fn is_repl(&self) -> bool {
        self.kind.is_repl()
    }
    pub const fn id(&self) -> u64 {
        self.id
    }
    pub fn dir(&self) -> PathBuf {
        self.kind.dir()
    }
    pub fn project_root(&self) -> Option<PathBuf> {
        self.kind.project_root()
    }
    pub fn enclosed_name(&self) -> &str {
        self.kind.as_str()
    }
    pub fn lineno(&self) -> usize {
        GLOBAL_STDIN.lineno()
    }
    pub fn block_begin(&self) -> usize {
        GLOBAL_STDIN.block_begin()
    }
    pub fn set_block_begin(&self) {
        GLOBAL_STDIN.set_block_begin(self.lineno())
    }
    pub fn insert_whitespace(&self, whitespace: &str) {
        GLOBAL_STDIN.insert_whitespace(whitespace);
    }
    pub fn set_indent(&self, indent: usize) {
        GLOBAL_STDIN.set_indent(indent);
    }
    pub fn file_stem(&self) -> String {
        match &self.kind {
            InputKind::File(filename) => filename
                .file_stem()
                .and_then(|f| f.to_str())
                .unwrap_or("_")
                .trim_end_matches(".d")
                .to_string(),
            InputKind::REPL | InputKind::Pipe(_) => "<stdin>".to_string(),
            InputKind::DummyREPL(stdin) => format!("<stdin_{}>", stdin.name),
            InputKind::Str(_) => "<string>".to_string(),
            InputKind::Dummy => "<dummy>".to_string(),
        }
    }
    pub fn full_path(&self) -> PathBuf {
        match &self.kind {
            InputKind::File(filename) => filename.clone(),
            _ => PathBuf::from(self.file_stem()),
        }
    }
    pub fn filename(&self) -> String {
        match &self.kind {
            InputKind::File(filename) => filename
                .file_name()
                .and_then(|f| f.to_str())
                .unwrap_or("_")
                .trim_end_matches(".d")
                .to_string(),
            _ => self.file_stem(),
        }
    }
        pub fn path(&self) -> &Path {
        self.kind.path()
    }
    pub fn module_name(&self) -> String {
        match &self.kind {
            InputKind::File(filename) => {
                let file_stem = if filename.file_stem() == Some(OsStr::new("__init__"))
                    || filename.file_stem() == Some(OsStr::new("__init__.d"))
                {
                    filename.parent().and_then(|p| p.file_stem())
                } else {
                    filename.file_stem()
                };
                file_stem
                    .and_then(|f| f.to_str())
                    .unwrap_or("_")
                    .trim_end_matches(".d")
                    .to_string()
            }
            InputKind::REPL | InputKind::Pipe(_) => "<stdin>".to_string(),
            InputKind::DummyREPL(stdin) => stdin.name.clone(),
            InputKind::Str(_) => "<string>".to_string(),
            InputKind::Dummy => "<dummy>".to_string(),
        }
    }
    pub fn read(&mut self) -> String {
        match &mut self.kind {
            InputKind::File(filename) => match VFS.read(filename.as_path()) {
                Ok(s) => s,
                Err(e) => {
                    let code = e.raw_os_error().unwrap_or(1);
                    println!(
                        "cannot read '{}': [Errno {code}] {e}",
                        filename.to_string_lossy()
                    );
                    process::exit(code);
                }
            },
            InputKind::Pipe(s) | InputKind::Str(s) => s.clone(),
            InputKind::REPL => GLOBAL_STDIN.read(),
            InputKind::DummyREPL(dummy) => dummy.read_line(),
            InputKind::Dummy => panic!("cannot read from a dummy file"),
        }
    }
    pub fn source_exists(&self) -> bool {
        match &self.kind {
            InputKind::File(filename) => filename.exists(),
            InputKind::Dummy => false,
            _ => true,
        }
    }
    pub fn try_read(&mut self) -> std::io::Result<String> {
        match &mut self.kind {
            InputKind::File(filename) => VFS.read(filename),
            InputKind::Pipe(s) | InputKind::Str(s) => Ok(s.clone()),
            InputKind::REPL => Ok(GLOBAL_STDIN.read()),
            InputKind::DummyREPL(dummy) => Ok(dummy.read_line()),
            InputKind::Dummy => panic!("cannot read from a dummy file"),
        }
    }
    pub fn read_non_dummy(&self) -> String {
        match &self.kind {
            InputKind::File(filename) => match VFS.read(filename) {
                Ok(s) => s,
                Err(e) => {
                    let code = e.raw_os_error().unwrap_or(1);
                    println!(
                        "cannot read '{}': [Errno {code}] {e}",
                        filename.to_string_lossy()
                    );
                    process::exit(code);
                }
            },
            InputKind::Pipe(s) | InputKind::Str(s) => s.clone(),
            InputKind::REPL => GLOBAL_STDIN.read(),
            InputKind::Dummy | InputKind::DummyREPL(_) => panic!("cannot read from a dummy file"),
        }
    }
    pub fn reread_lines(&self, ln_begin: usize, ln_end: usize) -> Vec<String> {
        power_assert!(ln_begin, >=, 1);
        match &self.kind {
            InputKind::File(filename) => match VFS.read(filename) {
                Ok(code) => {
                    let mut codes = vec![];
                    let mut lines = code.lines().map(ToString::to_string).skip(ln_begin - 1);
                    for _ in ln_begin..=ln_end {
                        codes.push(lines.next().unwrap_or("".to_string()));
                    }
                    codes
                }
                Err(_) => vec!["<file not found>".into()],
            },
            InputKind::Pipe(s) | InputKind::Str(s) => s
                .split('\n')
                .collect::<Vec<_>>()
                .get(ln_begin - 1..=ln_end - 1)
                .unwrap_or_default()
                .iter()
                .map(|s| s.to_string())
                .collect(),
            InputKind::REPL => {
                let block_begin = self.block_begin().saturating_sub(1);
                GLOBAL_STDIN.reread_lines(ln_begin + block_begin, ln_end + block_begin)
            }
            InputKind::DummyREPL(dummy) => dummy.reread_lines(ln_begin, ln_end),
            InputKind::Dummy => panic!("cannot read lines from a dummy file"),
        }
    }
    pub fn reread(&self) -> String {
        match &self.kind {
            InputKind::File(path) => VFS.read(path).unwrap(),
            InputKind::Pipe(s) | InputKind::Str(s) => s.clone(),
            InputKind::REPL => GLOBAL_STDIN.reread().trim_end().to_owned(),
            InputKind::DummyREPL(dummy) => dummy.reread().unwrap_or_default(),
            InputKind::Dummy => panic!("cannot read from a dummy file"),
        }
    }
                fn resolve_local(&self, path: &Path) -> Result<PathBuf, std::io::Error> {
        let mut dir = self.dir();
        dir.push(path);
        dir.set_extension("er");         let path = dir.canonicalize().or_else(|_| {
            dir.pop();             dir.push(path.iter().last().unwrap_or_default());             dir.push("__init__.er");             dir.canonicalize()
        })?;
        Ok(normalize_path(path))
    }
    fn resolve_local_decl(&self, dir: PathBuf, path: &Path) -> Result<PathBuf, std::io::Error> {
        self._resolve_local_decl(dir.clone(), path).or_else(|_| {
            let path = add_postfix_foreach(path, ".d");
            self._resolve_local_decl(dir, &path)
        })
    }
                        fn _resolve_local_decl(
        &self,
        mut dir: PathBuf,
        path: &Path,
    ) -> Result<PathBuf, std::io::Error> {
        if path == Path::new("") {
            let path = dir
                .join("__init__.d.er")
                .canonicalize()
                .or_else(|_| dir.join("__pycache__").join("__init__.d.er").canonicalize())
                .or_else(|_| dir.canonicalize())?;
            return Ok(path);
        }
        let mut comps = path.components();
        let last = comps
            .next_back()
            .ok_or_else(|| std::io::Error::new(std::io::ErrorKind::NotFound, "path is empty"))?;
        let last_path = Path::new(&last);
        dir.push(comps);
        dir.push(last_path);
        dir.set_extension("d.er");         let path = dir
            .canonicalize()
            .or_else(|_| {
                dir.pop();                 dir.push(last_path);                 dir.push("__init__.d.er");                 dir.canonicalize()
            })
            .or_else(|_| {
                dir.pop();                 dir.pop();                 dir.push("__pycache__");                 dir.push(last_path);                 dir.set_extension("d.er");                 dir.canonicalize()
            })
            .or_else(|_| {
                dir.pop();                 dir.pop();                 dir.push(last_path);                 dir.push("__pycache__");                 dir.push("__init__.d.er");                 dir.canonicalize()
            })?;
        Ok(normalize_path(path))
    }
    fn resolve_local_py(&self, path: &Path) -> Result<PathBuf, std::io::Error> {
        let mut dir = self.dir();
        dir.push(path);
        dir.set_extension("py");
        let path = dir.canonicalize().or_else(|_| {
            dir.pop();
            dir.push(path);
            dir.push("__init__.py");             dir.canonicalize()
        })?;
        Ok(normalize_path(path))
    }
    pub fn resolve_py(&self, path: &Path) -> Result<PathBuf, std::io::Error> {
        if let Ok(path) = self.resolve_local_py(path) {
            return Ok(path);
        }
        for sys_path in python_sys_path() {
            let mut dir = sys_path.clone();
            dir.push(path);
            dir.set_extension("py");
            if dir.exists() {
                return Ok(normalize_path(dir));
            }
            dir.pop();
            dir.push(path);
            dir.push("__init__.py");
            if dir.exists() {
                return Ok(normalize_path(dir));
            }
            if !EXPERIMENTAL_MODE {
                break;
            }
        }
        for pkgs_path in python_site_packages() {
            let mut dir = pkgs_path.clone();
            dir.push(path);
            dir.set_extension("py");
            if dir.exists() {
                return Ok(normalize_path(dir));
            }
            dir.pop();
            dir.push(path);
            dir.push("__init__.py");
            if dir.exists() {
                return Ok(normalize_path(dir));
            }
        }
        Err(std::io::Error::new(
            std::io::ErrorKind::NotFound,
            format!("cannot find module `{}`", path.display()),
        ))
    }
    pub fn resolve_path(&self, path: &Path, cfg: &ErgConfig) -> Option<PathBuf> {
        self.resolve_real_path(path, cfg)
            .or_else(|| self.resolve_decl_path(path, cfg))
    }
                            pub fn resolve_real_path(&self, path: &Path, cfg: &ErgConfig) -> Option<PathBuf> {
        if let Ok(path) = self.resolve_local(path) {
            Some(path)
        } else if let Ok(path) = erg_std_path()
            .join(format!("{}.er", path.display()))
            .canonicalize()
        {
            Some(normalize_path(path))
        } else if let Ok(path) = erg_std_path()
            .join(format!("{}", path.display()))
            .join("__init__.er")
            .canonicalize()
        {
            Some(normalize_path(path))
        } else if let Some(pkg) = self.resolve_project_dep_path(path, cfg, false) {
            Some(normalize_path(pkg))
        } else if path == Path::new("unsound") {
            Some(PathBuf::from("unsound"))
        } else {
            None
        }
    }
    fn resolve_project_dep_path(
        &self,
        path: &Path,
        cfg: &ErgConfig,
        decl: bool,
    ) -> Option<PathBuf> {
        let name = path.components().next()?.as_os_str();
        let pkg = cfg.packages.iter().find(|p| p.as_name == name)?;
        let root_path = if let Some(path) = pkg.path {
            PathBuf::from(path).canonicalize().ok()?
        } else {
            erg_pkgs_path().join(pkg.name).join(pkg.version)
        };
        if path.components().count() <= 1 {
            let full_path = if decl {
                root_path.join("src").join("lib.d.er")
            } else {
                root_path.join("src").join("lib.er")
            };
            full_path.canonicalize().ok()
        } else {
            let full_path = if decl {
                let path =
                    add_postfix_foreach(path.components().skip(1).collect::<PathBuf>(), ".d");
                root_path.join("src").join(path).with_extension("d.er")
            } else {
                let path = path.components().skip(1).collect::<PathBuf>();
                root_path.join("src").join(path).with_extension("er")
            };
            full_path.canonicalize().ok().or_else(|| {
                let full_path = if decl {
                    let path =
                        add_postfix_foreach(path.components().skip(1).collect::<PathBuf>(), ".d");
                    root_path.join("src").join(path).join("__init__.d.er")
                } else {
                    let path = path.components().skip(1).collect::<PathBuf>();
                    root_path.join("src").join(path).join("__init__.er")
                };
                full_path.canonicalize().ok()
            })
        }
    }
                                                        pub fn resolve_decl_path(&self, path: &Path, cfg: &ErgConfig) -> Option<PathBuf> {
        if let Ok(path) = self.resolve_local_decl(self.dir(), path) {
            return Some(path);
        }
                                                                if let Some((root, first)) = self.project_root().zip(path.components().next()) {
            if root.ends_with(first) || remove_postfix(root.clone(), ".d").ends_with(first) {
                let path = path.iter().skip(1).collect::<PathBuf>();
                if let Ok(path) = self.resolve_local_decl(root, &path) {
                    return Some(path);
                }
            }
        }
        if let Some(path) = Self::resolve_std_decl_path(erg_pystd_path(), path) {
            return Some(path);
        }
        if let Some(pkg) = self.resolve_project_dep_path(path, cfg, true) {
            return Some(normalize_path(pkg));
        }
        for site_packages in python_site_packages() {
            if let Some(path) = Self::resolve_site_pkgs_decl_path(site_packages, path) {
                return Some(path);
            }
        }
        if PYTHON_MODE {
            if let Ok(path) = self.resolve_py(path) {
                return Some(path);
            }
        }
        None
    }
            fn resolve_std_decl_path(root: &Path, path: &Path) -> Option<PathBuf> {
        let mut path = add_postfix_foreach(path, ".d");
        path.set_extension("d.er");         if let Ok(path) = root.join(&path).canonicalize() {
            Some(normalize_path(path))
                } else if let Ok(path) = root
            .join({
                path.set_extension("");
                path
            })
            .join("__init__.d.er")
            .canonicalize()
        {
            Some(normalize_path(path))
        } else {
            None
        }
    }
                        fn resolve_site_pkgs_decl_path(site_packages: &Path, path: &Path) -> Option<PathBuf> {
        let dir = path.parent().unwrap_or_else(|| Path::new(""));
        let mut file_path = PathBuf::from(path.file_stem().unwrap_or_default());
        file_path.set_extension("d.er");         if let Ok(path) = site_packages
            .join(dir)
            .join("__pycache__")
            .join(&file_path)
            .canonicalize()
        {
            Some(normalize_path(path))
        } else if let Ok(path) = site_packages
            .join(path)
            .join("__pycache__")
            .join("__init__.d.er")
            .canonicalize()
        {
            Some(normalize_path(path))
        } else {
            None
        }
    }
    pub fn try_push_path(mut path: PathBuf, add: &Path) -> Result<PathBuf, String> {
        if path.ends_with("__init__.d.er") {
            path.pop();
        }
        if let Ok(path) = path.join(add).canonicalize() {
            Ok(normalize_path(path))
        } else if let Ok(path) = path.join(format!("{}.d.er", add.display())).canonicalize() {
            Ok(normalize_path(path))
        } else if let Ok(path) = path
            .join(format!("{}.d", add.display()))
            .join("__init__.d.er")
            .canonicalize()
        {
            Ok(normalize_path(path))
        } else {
            Err(format!("{} // {}", path.display(), add.display()))
        }
    }
    pub fn decl_file_is(&self, decl_path: &Path) -> bool {
        let mut py_path = self.path().to_path_buf();
        py_path.set_extension("d.er");
        if decl_path == py_path {
            return true;
        }
        let last = py_path.file_name().unwrap_or_default().to_os_string();
        py_path.pop();
        py_path.push("__pycache__");
        py_path.push(last);
        decl_path == py_path
    }
    pub fn mode(&self) -> &'static str {
        if self.path().to_string_lossy().ends_with(".d.er") {
            "declare"
        } else {
            "exec"
        }
    }
}
#[derive(Debug)]
pub enum Output {
    Stdout(Stdout),
    File(File, String),
    Null,
}
impl Clone for Output {
    fn clone(&self) -> Self {
        match self {
            Self::Null => Self::Null,
            Self::Stdout(_) => Self::stdout(),
            Self::File(_, filename) => Self::file(filename.clone()),
        }
    }
}
impl std::io::Write for Output {
    fn write(&mut self, buf: &[u8]) -> std::io::Result<usize> {
        match self {
            Output::Stdout(stdout) => stdout.write(buf),
            Output::File(file, _) => file.write(buf),
            Output::Null => Ok(buf.len()),
        }
    }
    fn flush(&mut self) -> std::io::Result<()> {
        match self {
            Output::Stdout(stdout) => stdout.flush(),
            Output::File(file, _) => file.flush(),
            Output::Null => Ok(()),
        }
    }
}
impl From<Output> for Stdio {
    fn from(output: Output) -> Self {
        match output {
            Output::Stdout(_stdout) => Stdio::inherit(),
            Output::File(file, _) => Stdio::from(file),
            Output::Null => Stdio::null(),
        }
    }
}
impl Output {
    pub fn stdout() -> Self {
        Self::Stdout(std::io::stdout())
    }
    pub fn file(filename: String) -> Self {
        Self::File(File::open(&filename).unwrap(), filename)
    }
}
pub fn lsp_log(file: &str, line: u32, msg: &str) {
    let file_path = erg_path().join("els.log");
    let Ok(mut f) = (if file_path.exists() {
        File::options().append(true).open(file_path)
    } else {
        File::create(file_path)
    }) else {
        return;
    };
    let _ = f.write_all(format!("{file}@{line}: {msg}\n").as_bytes());
}
#[macro_export]
macro_rules! lsp_log {
    ($($arg:tt)*) => {
        let msg = format!($($arg)*);
        $crate::io::lsp_log(file!(), line!(), &msg);
    };
}