// Copyright 2013 The Rust Project Developers. See the COPYRIGHT // file at the top-level directory of this distribution and at // http://rust-lang.org/COPYRIGHT. // // Licensed under the Apache License, Version 2.0 or the MIT license // , at your // option. This file may not be copied, modified, or distributed // except according to those terms. /*! Synchronous File I/O This module provides a set of functions and traits for working with regular files & directories on a filesystem. At the top-level of the module are a set of freestanding functions, associated with various filesystem operations. They all operate on a `PathLike` object. All operations in this module, including those as part of `FileStream` et al block the task during execution. Most will raise `std::rt::io::{io_error,read_error}` conditions in the event of failure. Also included in this module are the `FileInfo` and `DirectoryInfo` traits. When `use`'d alongside a value whose type implements them (A `std::path::Path` impl is a part of this module), they expose a set of functions for operations against a given file location, depending on whether the path already exists. Whenever possible, the `{FileInfo, DirectoryInfo}` preserve the same semantics as their free function counterparts. */ use prelude::*; use super::support::PathLike; use super::{Reader, Writer, Seek}; use super::{SeekStyle, Read, Write}; use rt::rtio::{RtioFileStream, IoFactory, IoFactoryObject}; use rt::io::{io_error, read_error, EndOfFile, FileMode, FileAccess, FileStat, IoError, PathAlreadyExists, PathDoesntExist, MismatchedFileTypeForOperation, ignore_io_error}; use rt::local::Local; use option::{Some, None}; use path::Path; /// Open a file for reading/writing, as indicated by `path`. /// /// # Example /// /// use std; /// use std::path::Path; /// use std::rt::io::support::PathLike; /// use std::rt::io::file::open; /// use std::rt::io::{FileMode, FileAccess}; /// /// let p = &Path("/some/file/path.txt"); /// /// do io_error::cond.trap(|_| { /// // hoo-boy... /// }).inside { /// let stream = match open(p, Create, ReadWrite) { /// Some(s) => s, /// None => fail!("whoops! I'm sure this raised, anyways.."); /// } /// // do some stuff with that stream /// /// // the file stream will be closed at the end of this block /// } /// // .. /// /// `FileMode` and `FileAccess` provide information about the permissions /// context in which a given stream is created. More information about them /// can be found in `std::rt::io`'s docs. /// /// Note that, with this function, a `FileStream` is returned regardless of /// the access-limitations indicated by `FileAccess` (e.g. calling `write` on a /// `FileStream` opened as `ReadOnly` will raise an `io_error` condition at runtime). If you /// desire a more-correctly-constrained interface to files, use the /// `{open_stream, open_reader, open_writer}` methods that are a part of `FileInfo` /// /// # Errors /// /// This function will raise an `io_error` condition under a number of different circumstances, /// to include but not limited to: /// /// * Opening a file that already exists with `FileMode` of `Create` or vice versa (e.g. /// opening a non-existant file with `FileMode` or `Open`) /// * Attempting to open a file with a `FileAccess` that the user lacks permissions /// for /// * Filesystem-level errors (full disk, etc) pub fn open(path: &P, mode: FileMode, access: FileAccess ) -> Option { let open_result = unsafe { let io: *mut IoFactoryObject = Local::unsafe_borrow(); (*io).fs_open(path, mode, access) }; match open_result { Ok(fd) => Some(FileStream { fd: fd, last_nread: -1 }), Err(ioerr) => { io_error::cond.raise(ioerr); None } } } /// Unlink a file from the underlying filesystem. /// /// # Example /// /// use std; /// use std::path::Path; /// use std::rt::io::support::PathLike; /// use std::rt::io::file::unlink; /// /// let p = &Path("/some/file/path.txt"); /// unlink(p); /// // if we made it here without failing, then the /// // unlink operation was successful /// /// Note that, just because an unlink call was successful, it is not /// guaranteed that a file is immediately deleted (e.g. depending on /// platform, other open file descriptors may prevent immediate removal) /// /// # Errors /// /// This function will raise an `io_error` condition if the user lacks permissions to /// remove the file or if some other filesystem-level error occurs pub fn unlink(path: &P) { let unlink_result = unsafe { let io: *mut IoFactoryObject = Local::unsafe_borrow(); (*io).fs_unlink(path) }; match unlink_result { Ok(_) => (), Err(ioerr) => { io_error::cond.raise(ioerr); } } } /// Create a new, empty directory at the provided path /// /// # Example /// /// use std; /// use std::path::Path; /// use std::rt::io::support::PathLike; /// use std::rt::io::file::mkdir; /// /// let p = &Path("/some/dir"); /// mkdir(p); /// // If we got here, our directory exists! Horray! /// /// # Errors /// /// This call will raise an `io_error` condition if the user lacks permissions to make a /// new directory at the provided path, or if the directory already exists pub fn mkdir(path: &P) { let mkdir_result = unsafe { let io: *mut IoFactoryObject = Local::unsafe_borrow(); (*io).fs_mkdir(path) }; match mkdir_result { Ok(_) => (), Err(ioerr) => { io_error::cond.raise(ioerr); } } } /// Remove an existing, empty directory /// /// # Example /// /// use std; /// use std::path::Path; /// use std::rt::io::support::PathLike; /// use std::rt::io::file::rmdir; /// /// let p = &Path("/some/dir"); /// rmdir(p); /// // good riddance, you mean ol' directory /// /// # Errors /// /// This call will raise an `io_error` condition if the user lacks permissions to remove the /// directory at the provided path, or if the directory isn't empty pub fn rmdir(path: &P) { let rmdir_result = unsafe { let io: *mut IoFactoryObject = Local::unsafe_borrow(); (*io).fs_rmdir(path) }; match rmdir_result { Ok(_) => (), Err(ioerr) => { io_error::cond.raise(ioerr); } } } /// Get information on the file, directory, etc at the provided path /// /// Given a `rt::io::support::PathLike`, query the file system to get /// information about a file, directory, etc. /// /// Returns a `Some(std::rt::io::PathInfo)` on success /// /// # Example /// /// use std; /// use std::path::Path; /// use std::rt::io::support::PathLike; /// use std::rt::io::file::stat; /// /// let p = &Path("/some/file/path.txt"); /// /// do io_error::cond.trap(|_| { /// // hoo-boy... /// }).inside { /// let info = match stat(p) { /// Some(s) => s, /// None => fail!("whoops! I'm sure this raised, anyways.."); /// } /// if stat.is_file { /// // just imagine the possibilities ... /// } /// /// // the file stream will be closed at the end of this block /// } /// // .. /// /// # Errors /// /// This call will raise an `io_error` condition if the user lacks the requisite /// permissions to perform a `stat` call on the given path or if there is no /// entry in the filesystem at the provided path. pub fn stat(path: &P) -> Option { let open_result = unsafe { let io: *mut IoFactoryObject = Local::unsafe_borrow(); (*io).fs_stat(path) }; match open_result { Ok(p) => { Some(p) }, Err(ioerr) => { io_error::cond.raise(ioerr); None } } } /// Retrieve a vector containing all entries within a provided directory /// /// # Example /// /// use std; /// use std::path::Path; /// use std::rt::io::support::PathLike; /// use std::rt::io::file::readdir; /// /// fn visit_dirs(dir: &Path, cb: &fn(&Path)) { /// if dir.is_dir() { /// let contents = dir.readdir(); /// for entry in contents.iter() { /// if entry.is_dir() { visit_dirs(entry, cb); } /// else { cb(entry); } /// } /// } /// else { fail!("nope"); } /// } /// /// # Errors /// /// Will raise an `io_error` condition if the provided `path` doesn't exist, /// the process lacks permissions to view the contents or if the `path` points /// at a non-directory file pub fn readdir(path: &P) -> Option<~[Path]> { let readdir_result = unsafe { let io: *mut IoFactoryObject = Local::unsafe_borrow(); (*io).fs_readdir(path, 0) }; match readdir_result { Ok(p) => { Some(p) }, Err(ioerr) => { io_error::cond.raise(ioerr); None } } } /// Constrained version of `FileStream` that only exposes read-specific operations. /// /// Can be retreived via `FileInfo.open_reader()`. pub struct FileReader { priv stream: FileStream } /// a `std::rt::io::Reader` trait impl for file I/O. impl Reader for FileReader { fn read(&mut self, buf: &mut [u8]) -> Option { self.stream.read(buf) } fn eof(&mut self) -> bool { self.stream.eof() } } /// a `std::rt::io::Seek` trait impl for file I/O. impl Seek for FileReader { fn tell(&self) -> u64 { self.stream.tell() } fn seek(&mut self, pos: i64, style: SeekStyle) { self.stream.seek(pos, style); } } /// Constrained version of `FileStream` that only exposes write-specific operations. /// /// Can be retreived via `FileInfo.open_writer()`. pub struct FileWriter { priv stream: FileStream } /// a `std::rt::io::Writer` trait impl for file I/O. impl Writer for FileWriter { fn write(&mut self, buf: &[u8]) { self.stream.write(buf); } fn flush(&mut self) { self.stream.flush(); } } /// a `std::rt::io::Seek` trait impl for file I/O. impl Seek for FileWriter { fn tell(&self) -> u64 { self.stream.tell() } fn seek(&mut self, pos: i64, style: SeekStyle) { self.stream.seek(pos, style); } } /// Unconstrained file access type that exposes read and write operations /// /// Can be retreived via `file::open()` and `FileInfo.open_stream()`. /// /// # Errors /// /// This type will raise an io_error condition if operations are attempted against /// it for which its underlying file descriptor was not configured at creation /// time, via the `FileAccess` parameter to `file::open()`. /// /// For this reason, it is best to use the access-constrained wrappers that are /// exposed via `FileInfo.open_reader()` and `FileInfo.open_writer()`. pub struct FileStream { fd: ~RtioFileStream, last_nread: int, } /// a `std::rt::io::Reader` trait impl for file I/O. impl Reader for FileStream { fn read(&mut self, buf: &mut [u8]) -> Option { match self.fd.read(buf) { Ok(read) => { self.last_nread = read; match read { 0 => None, _ => Some(read as uint) } }, Err(ioerr) => { // EOF is indicated by returning None if ioerr.kind != EndOfFile { read_error::cond.raise(ioerr); } return None; } } } fn eof(&mut self) -> bool { self.last_nread == 0 } } /// a `std::rt::io::Writer` trait impl for file I/O. impl Writer for FileStream { fn write(&mut self, buf: &[u8]) { match self.fd.write(buf) { Ok(_) => (), Err(ioerr) => { io_error::cond.raise(ioerr); } } } fn flush(&mut self) { match self.fd.flush() { Ok(_) => (), Err(ioerr) => { read_error::cond.raise(ioerr); } } } } /// a `std::rt::io:Seek` trait impl for file I/O. impl Seek for FileStream { fn tell(&self) -> u64 { let res = self.fd.tell(); match res { Ok(cursor) => cursor, Err(ioerr) => { read_error::cond.raise(ioerr); return -1; } } } fn seek(&mut self, pos: i64, style: SeekStyle) { match self.fd.seek(pos, style) { Ok(_) => { // successful seek resets EOF indicator self.last_nread = -1; () }, Err(ioerr) => { read_error::cond.raise(ioerr); } } } } /// Shared functionality between `FileInfo` and `DirectoryInfo` pub trait FileSystemInfo { /// Get the filesystem path that this instance points at, /// whether it is valid or not. In this way, it can be used to /// to specify a path of a non-existent file which it /// later creates fn get_path<'a>(&'a self) -> &'a Path; /// Get information on the file, directory, etc at the provided path /// /// Consult the `file::stat` documentation for more info. /// /// This call preserves identical runtime/error semantics with `file::stat` fn stat(&self) -> Option { stat(self.get_path()) } /// Boolean value indicator whether the underlying file exists on the filesystem /// /// # Errors /// /// Will not raise a condition fn exists(&self) -> bool { match ignore_io_error(|| self.stat()) { Some(_) => true, None => false } } } /// Represents a file, whose underlying path may or may not be valid /// /// # Example /// /// * Check if a file exists, reading from it if so /// /// ```rust /// use std; /// use std::path::Path; /// use std::rt::io::file::{FileInfo, FileReader}; /// /// let f = &Path("/some/file/path.txt"); /// if f.exists() { /// let reader = f.open_reader(Open); /// let mut mem = [0u8, 8*64000]; /// reader.read(mem); /// // ... /// } /// ``` /// /// * Is the given path a file? /// /// ```rust /// let f = get_file_path_from_wherever(); /// match f.is_file() { /// true => doing_something_with_a_file(f), /// _ => {} /// } /// ``` pub trait FileInfo : FileSystemInfo { /// Whether the underlying implemention (be it a file path, /// or something else) points at a "regular file" on the FS. Will return /// false for paths to non-existent locations or directories or /// other non-regular files (named pipes, etc). /// /// # Errors /// /// Will not raise a condition fn is_file(&self) -> bool { match ignore_io_error(|| self.stat()) { Some(s) => s.is_file, None => false } } /// Attempts to open a regular file for reading/writing based /// on provided inputs /// /// See `file::open` for more information on runtime semantics and error conditions fn open_stream(&self, mode: FileMode, access: FileAccess) -> Option { match ignore_io_error(|| self.stat()) { Some(s) => match s.is_file { true => open(self.get_path(), mode, access), false => None }, None => open(self.get_path(), mode, access) } } /// Attempts to open a regular file in read-only mode, based /// on provided inputs /// /// See `file::open` for more information on runtime semantics and error conditions fn open_reader(&self, mode: FileMode) -> Option { match self.open_stream(mode, Read) { Some(s) => Some(FileReader { stream: s}), None => None } } /// Attempts to open a regular file in write-only mode, based /// on provided inputs /// /// See `file::open` for more information on runtime semantics and error conditions fn open_writer(&self, mode: FileMode) -> Option { match self.open_stream(mode, Write) { Some(s) => Some(FileWriter { stream: s}), None => None } } /// Attempt to remove a file from the filesystem /// /// See `file::unlink` for more information on runtime semantics and error conditions fn unlink(&self) { unlink(self.get_path()); } } /// `FileSystemInfo` implementation for `Path`s impl FileSystemInfo for Path { fn get_path<'a>(&'a self) -> &'a Path { self } } /// `FileInfo` implementation for `Path`s impl FileInfo for Path { } /// Represents a directory, whose underlying path may or may not be valid /// /// # Example /// /// * Check if a directory exists, `mkdir`'ing it if not /// /// ```rust /// use std; /// use std::path::Path; /// use std::rt::io::file::{DirectoryInfo}; /// /// let dir = &Path("/some/dir"); /// if !dir.exists() { /// dir.mkdir(); /// } /// ``` /// /// * Is the given path a directory? If so, iterate on its contents /// /// ```rust /// fn visit_dirs(dir: &Path, cb: &fn(&Path)) { /// if dir.is_dir() { /// let contents = dir.readdir(); /// for entry in contents.iter() { /// if entry.is_dir() { visit_dirs(entry, cb); } /// else { cb(entry); } /// } /// } /// else { fail!("nope"); } /// } /// ``` pub trait DirectoryInfo : FileSystemInfo { /// Whether the underlying implemention (be it a file path, /// or something else) is pointing at a directory in the underlying FS. /// Will return false for paths to non-existent locations or if the item is /// not a directory (eg files, named pipes, links, etc) /// /// # Errors /// /// Will not raise a condition fn is_dir(&self) -> bool { match ignore_io_error(|| self.stat()) { Some(s) => s.is_dir, None => false } } /// Create a directory at the location pointed to by the /// type underlying the given `DirectoryInfo`. /// /// # Errors /// /// This method will raise a `PathAlreadyExists` kind of `io_error` condition /// if the provided path exists /// /// See `file::mkdir` for more information on runtime semantics and error conditions fn mkdir(&self) { match ignore_io_error(|| self.stat()) { Some(_) => { let path = self.get_path(); io_error::cond.raise(IoError { kind: PathAlreadyExists, desc: "Path already exists", detail: Some(format!("{} already exists; can't mkdir it", path.display())) }) }, None => mkdir(self.get_path()) } } /// Remove a directory at the given location. /// /// # Errors /// /// This method will raise a `PathDoesntExist` kind of `io_error` condition /// if the provided path exists. It will raise a `MismatchedFileTypeForOperation` /// kind of `io_error` condition if the provided path points at any /// non-directory file type /// /// See `file::rmdir` for more information on runtime semantics and error conditions fn rmdir(&self) { match ignore_io_error(|| self.stat()) { Some(s) => { match s.is_dir { true => rmdir(self.get_path()), false => { let path = self.get_path(); let ioerr = IoError { kind: MismatchedFileTypeForOperation, desc: "Cannot do rmdir() on a non-directory", detail: Some(format!( "{} is a non-directory; can't rmdir it", path.display())) }; io_error::cond.raise(ioerr); } } }, None => { let path = self.get_path(); io_error::cond.raise(IoError { kind: PathDoesntExist, desc: "Path doesn't exist", detail: Some(format!("{} doesn't exist; can't rmdir it", path.display())) }) } } } // Get a collection of all entries at the given // directory fn readdir(&self) -> Option<~[Path]> { readdir(self.get_path()) } } /// `DirectoryInfo` impl for `path::Path` impl DirectoryInfo for Path { } #[cfg(test)] mod test { use super::super::{SeekSet, SeekCur, SeekEnd, io_error, Read, Create, Open, ReadWrite}; use super::super::super::test::*; use option::{Some, None}; use path::Path; use super::*; use iter::range; #[test] fn file_test_io_smoke_test() { do run_in_mt_newsched_task { let message = "it's alright. have a good time"; let filename = &Path::new("./tmp/file_rt_io_file_test.txt"); { let mut write_stream = open(filename, Create, ReadWrite).unwrap(); write_stream.write(message.as_bytes()); } { use str; let mut read_stream = open(filename, Open, Read).unwrap(); let mut read_buf = [0, .. 1028]; let read_str = match read_stream.read(read_buf).unwrap() { -1|0 => fail!("shouldn't happen"), n => str::from_utf8(read_buf.slice_to(n)) }; assert!(read_str == message.to_owned()); } unlink(filename); } } #[test] fn file_test_io_invalid_path_opened_without_create_should_raise_condition() { do run_in_mt_newsched_task { let filename = &Path::new("./tmp/file_that_does_not_exist.txt"); let mut called = false; do io_error::cond.trap(|_| { called = true; }).inside { let result = open(filename, Open, Read); assert!(result.is_none()); } assert!(called); } } #[test] fn file_test_iounlinking_invalid_path_should_raise_condition() { do run_in_mt_newsched_task { let filename = &Path::new("./tmp/file_another_file_that_does_not_exist.txt"); let mut called = false; do io_error::cond.trap(|_| { called = true; }).inside { unlink(filename); } assert!(called); } } #[test] fn file_test_io_non_positional_read() { do run_in_mt_newsched_task { use str; let message = "ten-four"; let mut read_mem = [0, .. 8]; let filename = &Path::new("./tmp/file_rt_io_file_test_positional.txt"); { let mut rw_stream = open(filename, Create, ReadWrite).unwrap(); rw_stream.write(message.as_bytes()); } { let mut read_stream = open(filename, Open, Read).unwrap(); { let read_buf = read_mem.mut_slice(0, 4); read_stream.read(read_buf); } { let read_buf = read_mem.mut_slice(4, 8); read_stream.read(read_buf); } } unlink(filename); let read_str = str::from_utf8(read_mem); assert!(read_str == message.to_owned()); } } #[test] fn file_test_io_seek_and_tell_smoke_test() { do run_in_mt_newsched_task { use str; let message = "ten-four"; let mut read_mem = [0, .. 4]; let set_cursor = 4 as u64; let mut tell_pos_pre_read; let mut tell_pos_post_read; let filename = &Path::new("./tmp/file_rt_io_file_test_seeking.txt"); { let mut rw_stream = open(filename, Create, ReadWrite).unwrap(); rw_stream.write(message.as_bytes()); } { let mut read_stream = open(filename, Open, Read).unwrap(); read_stream.seek(set_cursor as i64, SeekSet); tell_pos_pre_read = read_stream.tell(); read_stream.read(read_mem); tell_pos_post_read = read_stream.tell(); } unlink(filename); let read_str = str::from_utf8(read_mem); assert!(read_str == message.slice(4, 8).to_owned()); assert!(tell_pos_pre_read == set_cursor); assert!(tell_pos_post_read == message.len() as u64); } } #[test] fn file_test_io_seek_and_write() { do run_in_mt_newsched_task { use str; let initial_msg = "food-is-yummy"; let overwrite_msg = "-the-bar!!"; let final_msg = "foo-the-bar!!"; let seek_idx = 3; let mut read_mem = [0, .. 13]; let filename = &Path::new("./tmp/file_rt_io_file_test_seek_and_write.txt"); { let mut rw_stream = open(filename, Create, ReadWrite).unwrap(); rw_stream.write(initial_msg.as_bytes()); rw_stream.seek(seek_idx as i64, SeekSet); rw_stream.write(overwrite_msg.as_bytes()); } { let mut read_stream = open(filename, Open, Read).unwrap(); read_stream.read(read_mem); } unlink(filename); let read_str = str::from_utf8(read_mem); assert!(read_str == final_msg.to_owned()); } } #[test] fn file_test_io_seek_shakedown() { do run_in_mt_newsched_task { use str; // 01234567890123 let initial_msg = "qwer-asdf-zxcv"; let chunk_one = "qwer"; let chunk_two = "asdf"; let chunk_three = "zxcv"; let mut read_mem = [0, .. 4]; let filename = &Path::new("./tmp/file_rt_io_file_test_seek_shakedown.txt"); { let mut rw_stream = open(filename, Create, ReadWrite).unwrap(); rw_stream.write(initial_msg.as_bytes()); } { let mut read_stream = open(filename, Open, Read).unwrap(); read_stream.seek(-4, SeekEnd); read_stream.read(read_mem); let read_str = str::from_utf8(read_mem); assert!(read_str == chunk_three.to_owned()); read_stream.seek(-9, SeekCur); read_stream.read(read_mem); let read_str = str::from_utf8(read_mem); assert!(read_str == chunk_two.to_owned()); read_stream.seek(0, SeekSet); read_stream.read(read_mem); let read_str = str::from_utf8(read_mem); assert!(read_str == chunk_one.to_owned()); } unlink(filename); } } #[test] fn file_test_stat_is_correct_on_is_file() { do run_in_mt_newsched_task { let filename = &Path::new("./tmp/file_stat_correct_on_is_file.txt"); { let mut fs = open(filename, Create, ReadWrite).unwrap(); let msg = "hw"; fs.write(msg.as_bytes()); } let stat_res = match stat(filename) { Some(s) => s, None => fail!("shouldn't happen") }; assert!(stat_res.is_file); unlink(filename); } } #[test] fn file_test_stat_is_correct_on_is_dir() { do run_in_mt_newsched_task { let filename = &Path::new("./tmp/file_stat_correct_on_is_dir"); mkdir(filename); let stat_res = match stat(filename) { Some(s) => s, None => fail!("shouldn't happen") }; assert!(stat_res.is_dir); rmdir(filename); } } #[test] fn file_test_fileinfo_false_when_checking_is_file_on_a_directory() { do run_in_mt_newsched_task { let dir = &Path::new("./tmp/fileinfo_false_on_dir"); mkdir(dir); assert!(dir.is_file() == false); rmdir(dir); } } #[test] fn file_test_fileinfo_check_exists_before_and_after_file_creation() { do run_in_mt_newsched_task { let file = &Path::new("./tmp/fileinfo_check_exists_b_and_a.txt"); { let msg = "foo".as_bytes(); let mut w = file.open_writer(Create); w.write(msg); } assert!(file.exists()); file.unlink(); assert!(!file.exists()); } } #[test] fn file_test_directoryinfo_check_exists_before_and_after_mkdir() { do run_in_mt_newsched_task { let dir = &Path::new("./tmp/before_and_after_dir"); assert!(!dir.exists()); dir.mkdir(); assert!(dir.exists()); assert!(dir.is_dir()); dir.rmdir(); assert!(!dir.exists()); } } #[test] fn file_test_directoryinfo_readdir() { use str; do run_in_mt_newsched_task { let dir = &Path::new("./tmp/di_readdir"); dir.mkdir(); let prefix = "foo"; for n in range(0,3) { let f = dir.join(format!("{}.txt", n)); let mut w = f.open_writer(Create); let msg_str = (prefix + n.to_str().to_owned()).to_owned(); let msg = msg_str.as_bytes(); w.write(msg); } match dir.readdir() { Some(files) => { let mut mem = [0u8, .. 4]; for f in files.iter() { { let n = f.filestem_str(); let mut r = f.open_reader(Open); r.read(mem); let read_str = str::from_utf8(mem); let expected = match n { None|Some("") => fail!("really shouldn't happen.."), Some(n) => prefix+n }; assert!(expected == read_str); } f.unlink(); } }, None => fail!("shouldn't happen") } dir.rmdir(); } } }