2020-12-12 08:00:42 +00:00
|
|
|
// Copyright 2020 Google LLC
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
//
|
|
|
|
// https://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
|
2022-10-07 05:24:35 +00:00
|
|
|
use std::io::{Stderr, Stdout, Write};
|
2021-06-09 23:45:47 +00:00
|
|
|
use std::path::{Component, Path, PathBuf};
|
2022-06-08 02:43:12 +00:00
|
|
|
use std::str::FromStr;
|
2020-12-12 08:00:42 +00:00
|
|
|
use std::sync::{Mutex, MutexGuard};
|
2021-03-14 17:46:35 +00:00
|
|
|
use std::{fmt, io};
|
2020-12-12 08:00:42 +00:00
|
|
|
|
2021-06-03 05:02:06 +00:00
|
|
|
use atty::Stream;
|
2021-06-09 23:45:47 +00:00
|
|
|
use jujutsu_lib::repo_path::{RepoPath, RepoPathComponent, RepoPathJoin};
|
2021-05-15 16:16:31 +00:00
|
|
|
use jujutsu_lib::settings::UserSettings;
|
2020-12-12 08:00:42 +00:00
|
|
|
|
2022-10-07 03:52:01 +00:00
|
|
|
use crate::formatter::{Formatter, FormatterFactory};
|
2020-12-12 08:00:42 +00:00
|
|
|
|
2022-10-19 00:44:10 +00:00
|
|
|
pub struct Ui {
|
2020-12-12 08:00:42 +00:00
|
|
|
cwd: PathBuf,
|
2022-10-07 03:52:01 +00:00
|
|
|
formatter_factory: FormatterFactory,
|
2022-10-19 00:44:10 +00:00
|
|
|
output_pair: UiOutputPair,
|
2020-12-12 08:00:42 +00:00
|
|
|
settings: UserSettings,
|
|
|
|
}
|
|
|
|
|
2022-06-08 02:43:12 +00:00
|
|
|
#[derive(Clone, Copy, Debug, Eq, PartialEq)]
|
|
|
|
pub enum ColorChoice {
|
|
|
|
Always,
|
|
|
|
Never,
|
|
|
|
Auto,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Default for ColorChoice {
|
|
|
|
fn default() -> Self {
|
|
|
|
ColorChoice::Auto
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl FromStr for ColorChoice {
|
|
|
|
type Err = &'static str;
|
|
|
|
|
|
|
|
fn from_str(s: &str) -> Result<Self, Self::Err> {
|
|
|
|
match s {
|
|
|
|
"always" => Ok(ColorChoice::Always),
|
|
|
|
"never" => Ok(ColorChoice::Never),
|
|
|
|
"auto" => Ok(ColorChoice::Auto),
|
|
|
|
_ => Err("must be one of always, never, or auto"),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn color_setting(settings: &UserSettings) -> ColorChoice {
|
|
|
|
settings
|
2022-03-19 17:00:13 +00:00
|
|
|
.config()
|
|
|
|
.get_string("ui.color")
|
2022-06-08 02:43:12 +00:00
|
|
|
.ok()
|
|
|
|
.and_then(|s| s.parse().ok())
|
|
|
|
.unwrap_or_default()
|
|
|
|
}
|
|
|
|
|
2022-10-08 06:37:04 +00:00
|
|
|
fn use_color(choice: ColorChoice, maybe_tty: bool) -> bool {
|
2022-06-08 02:43:12 +00:00
|
|
|
match choice {
|
|
|
|
ColorChoice::Always => true,
|
|
|
|
ColorChoice::Never => false,
|
2022-10-08 06:37:04 +00:00
|
|
|
ColorChoice::Auto => maybe_tty && atty::is(Stream::Stdout),
|
2022-03-19 17:00:13 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-10-19 00:44:10 +00:00
|
|
|
impl Ui {
|
2020-12-12 08:00:42 +00:00
|
|
|
pub fn new(
|
|
|
|
cwd: PathBuf,
|
2022-10-19 00:44:10 +00:00
|
|
|
stdout: Box<dyn Write>,
|
|
|
|
stderr: Box<dyn Write>,
|
2021-06-03 05:02:06 +00:00
|
|
|
color: bool,
|
2020-12-12 08:00:42 +00:00
|
|
|
settings: UserSettings,
|
2022-10-19 00:44:10 +00:00
|
|
|
) -> Ui {
|
2022-10-07 03:52:01 +00:00
|
|
|
let formatter_factory = FormatterFactory::prepare(&settings, color);
|
2020-12-12 08:00:42 +00:00
|
|
|
Ui {
|
|
|
|
cwd,
|
2022-10-07 03:52:01 +00:00
|
|
|
formatter_factory,
|
2022-10-07 05:24:35 +00:00
|
|
|
output_pair: UiOutputPair::Dyn {
|
|
|
|
stdout: Mutex::new(stdout),
|
|
|
|
stderr: Mutex::new(stderr),
|
|
|
|
},
|
2020-12-12 08:00:42 +00:00
|
|
|
settings,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-10-19 00:44:10 +00:00
|
|
|
pub fn for_terminal(settings: UserSettings) -> Ui {
|
2020-12-12 08:00:42 +00:00
|
|
|
let cwd = std::env::current_dir().unwrap();
|
2022-10-08 06:37:04 +00:00
|
|
|
let color = use_color(color_setting(&settings), true);
|
2022-10-07 05:24:35 +00:00
|
|
|
let formatter_factory = FormatterFactory::prepare(&settings, color);
|
|
|
|
Ui {
|
|
|
|
cwd,
|
|
|
|
formatter_factory,
|
|
|
|
output_pair: UiOutputPair::Terminal {
|
|
|
|
stdout: io::stdout(),
|
|
|
|
stderr: io::stderr(),
|
|
|
|
},
|
|
|
|
settings,
|
|
|
|
}
|
2020-12-12 08:00:42 +00:00
|
|
|
}
|
|
|
|
|
2022-06-09 03:11:02 +00:00
|
|
|
/// Reconfigures the underlying outputs with the new color choice.
|
2022-10-08 06:37:04 +00:00
|
|
|
pub fn reset_color(&mut self, choice: ColorChoice) {
|
|
|
|
let maybe_tty = matches!(&self.output_pair, UiOutputPair::Terminal { .. });
|
|
|
|
let color = use_color(choice, maybe_tty);
|
2022-10-07 03:52:01 +00:00
|
|
|
if self.formatter_factory.is_color() != color {
|
2022-10-07 04:24:44 +00:00
|
|
|
self.formatter_factory = FormatterFactory::prepare(&self.settings, color);
|
2022-06-09 03:11:02 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-12-12 08:00:42 +00:00
|
|
|
pub fn cwd(&self) -> &Path {
|
|
|
|
&self.cwd
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn settings(&self) -> &UserSettings {
|
|
|
|
&self.settings
|
|
|
|
}
|
|
|
|
|
2022-10-07 11:57:35 +00:00
|
|
|
pub fn new_formatter<'output, W: Write + 'output>(
|
2021-06-03 04:43:00 +00:00
|
|
|
&self,
|
2022-10-07 11:57:35 +00:00
|
|
|
output: W,
|
2021-06-03 04:43:00 +00:00
|
|
|
) -> Box<dyn Formatter + 'output> {
|
2022-10-07 03:52:01 +00:00
|
|
|
self.formatter_factory.new_formatter(output)
|
2021-06-03 04:43:00 +00:00
|
|
|
}
|
|
|
|
|
2022-10-07 04:24:44 +00:00
|
|
|
/// Creates a formatter for the locked stdout stream.
|
|
|
|
///
|
|
|
|
/// Labels added to the returned formatter should be removed by caller.
|
|
|
|
/// Otherwise the last color would persist.
|
|
|
|
pub fn stdout_formatter<'a>(&'a self) -> Box<dyn Formatter + 'a> {
|
2022-10-07 05:24:35 +00:00
|
|
|
match &self.output_pair {
|
|
|
|
UiOutputPair::Dyn { stdout, .. } => {
|
|
|
|
let output = DynWriteLock(stdout.lock().unwrap());
|
|
|
|
self.new_formatter(output)
|
|
|
|
}
|
|
|
|
UiOutputPair::Terminal { stdout, .. } => self.new_formatter(stdout.lock()),
|
|
|
|
}
|
2022-04-07 06:25:01 +00:00
|
|
|
}
|
|
|
|
|
2022-10-07 04:24:44 +00:00
|
|
|
/// Creates a formatter for the locked stderr stream.
|
|
|
|
pub fn stderr_formatter<'a>(&'a self) -> Box<dyn Formatter + 'a> {
|
2022-10-07 05:24:35 +00:00
|
|
|
match &self.output_pair {
|
|
|
|
UiOutputPair::Dyn { stderr, .. } => {
|
|
|
|
let output = DynWriteLock(stderr.lock().unwrap());
|
|
|
|
self.new_formatter(output)
|
|
|
|
}
|
|
|
|
UiOutputPair::Terminal { stderr, .. } => self.new_formatter(stderr.lock()),
|
|
|
|
}
|
2020-12-12 08:00:42 +00:00
|
|
|
}
|
|
|
|
|
2021-04-07 06:05:16 +00:00
|
|
|
pub fn write(&mut self, text: &str) -> io::Result<()> {
|
2022-10-07 05:24:35 +00:00
|
|
|
let data = text.as_bytes();
|
|
|
|
match &mut self.output_pair {
|
|
|
|
UiOutputPair::Dyn { stdout, .. } => stdout.get_mut().unwrap().write_all(data),
|
|
|
|
UiOutputPair::Terminal { stdout, .. } => stdout.write_all(data),
|
|
|
|
}
|
2020-12-12 08:00:42 +00:00
|
|
|
}
|
|
|
|
|
2021-04-07 06:05:16 +00:00
|
|
|
pub fn write_fmt(&mut self, fmt: fmt::Arguments<'_>) -> io::Result<()> {
|
2022-10-07 05:24:35 +00:00
|
|
|
match &mut self.output_pair {
|
|
|
|
UiOutputPair::Dyn { stdout, .. } => stdout.get_mut().unwrap().write_fmt(fmt),
|
|
|
|
UiOutputPair::Terminal { stdout, .. } => stdout.write_fmt(fmt),
|
|
|
|
}
|
2020-12-12 08:00:42 +00:00
|
|
|
}
|
|
|
|
|
2022-05-02 15:06:44 +00:00
|
|
|
pub fn write_hint(&mut self, text: impl AsRef<str>) -> io::Result<()> {
|
|
|
|
let mut formatter = self.stderr_formatter();
|
2022-10-06 11:16:41 +00:00
|
|
|
formatter.add_label("hint")?;
|
2022-05-02 15:06:44 +00:00
|
|
|
formatter.write_str(text.as_ref())?;
|
|
|
|
formatter.remove_label()?;
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2022-05-02 19:58:32 +00:00
|
|
|
pub fn write_warn(&mut self, text: impl AsRef<str>) -> io::Result<()> {
|
|
|
|
let mut formatter = self.stderr_formatter();
|
2022-10-06 11:16:41 +00:00
|
|
|
formatter.add_label("warning")?;
|
2022-05-02 19:58:32 +00:00
|
|
|
formatter.write_str(text.as_ref())?;
|
|
|
|
formatter.remove_label()?;
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2021-04-07 06:05:16 +00:00
|
|
|
pub fn write_error(&mut self, text: &str) -> io::Result<()> {
|
2022-04-07 06:25:01 +00:00
|
|
|
let mut formatter = self.stderr_formatter();
|
2022-10-06 11:16:41 +00:00
|
|
|
formatter.add_label("error")?;
|
2021-06-02 22:50:08 +00:00
|
|
|
formatter.write_str(text)?;
|
|
|
|
formatter.remove_label()?;
|
2021-04-07 06:05:16 +00:00
|
|
|
Ok(())
|
2020-12-12 08:00:42 +00:00
|
|
|
}
|
|
|
|
|
2021-06-09 23:45:47 +00:00
|
|
|
/// Parses a path relative to cwd into a RepoPath relative to wc_path
|
|
|
|
pub fn parse_file_path(
|
|
|
|
&self,
|
|
|
|
wc_path: &Path,
|
|
|
|
input: &str,
|
|
|
|
) -> Result<RepoPath, FilePathParseError> {
|
|
|
|
let repo_relative_path = relative_path(wc_path, &self.cwd.join(input));
|
|
|
|
let mut repo_path = RepoPath::root();
|
|
|
|
for component in repo_relative_path.components() {
|
|
|
|
match component {
|
|
|
|
Component::Normal(a) => {
|
|
|
|
repo_path = repo_path.join(&RepoPathComponent::from(a.to_str().unwrap()));
|
|
|
|
}
|
|
|
|
Component::CurDir => {}
|
|
|
|
Component::ParentDir => {
|
|
|
|
if let Some(parent) = repo_path.parent() {
|
|
|
|
repo_path = parent;
|
|
|
|
} else {
|
|
|
|
return Err(FilePathParseError::InputNotInRepo(input.to_string()));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
_ => {
|
|
|
|
return Err(FilePathParseError::InputNotInRepo(input.to_string()));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Ok(repo_path)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-10-19 00:44:10 +00:00
|
|
|
enum UiOutputPair {
|
2022-10-07 05:24:35 +00:00
|
|
|
Dyn {
|
2022-10-19 00:44:10 +00:00
|
|
|
stdout: Mutex<Box<dyn Write>>,
|
|
|
|
stderr: Mutex<Box<dyn Write>>,
|
2022-10-07 05:24:35 +00:00
|
|
|
},
|
|
|
|
Terminal {
|
|
|
|
stdout: Stdout,
|
|
|
|
stderr: Stderr,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
2022-10-07 04:24:44 +00:00
|
|
|
/// Wrapper to implement `Write` for locked `Box<dyn Write>`.
|
2022-10-19 01:09:43 +00:00
|
|
|
struct DynWriteLock<'a, T>(MutexGuard<'a, T>);
|
2022-10-07 04:24:44 +00:00
|
|
|
|
2022-10-19 01:09:43 +00:00
|
|
|
impl<T: Write> Write for DynWriteLock<'_, T> {
|
2022-10-07 04:24:44 +00:00
|
|
|
fn write(&mut self, buf: &[u8]) -> io::Result<usize> {
|
|
|
|
self.0.write(buf)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn write_all(&mut self, buf: &[u8]) -> io::Result<()> {
|
|
|
|
self.0.write_all(buf)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn flush(&mut self) -> io::Result<()> {
|
|
|
|
self.0.flush()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-06-09 23:45:47 +00:00
|
|
|
#[derive(PartialEq, Eq, Clone, Debug)]
|
|
|
|
pub enum FilePathParseError {
|
|
|
|
InputNotInRepo(String),
|
2021-05-16 18:06:44 +00:00
|
|
|
}
|
|
|
|
|
2022-03-09 22:12:32 +00:00
|
|
|
pub fn relative_path(mut from: &Path, to: &Path) -> PathBuf {
|
2021-05-16 18:06:44 +00:00
|
|
|
let mut result = PathBuf::from("");
|
|
|
|
loop {
|
|
|
|
if let Ok(suffix) = to.strip_prefix(from) {
|
2022-04-20 05:15:38 +00:00
|
|
|
result = result.join(suffix);
|
|
|
|
break;
|
2021-05-16 18:06:44 +00:00
|
|
|
}
|
|
|
|
if let Some(parent) = from.parent() {
|
|
|
|
result = result.join("..");
|
|
|
|
from = parent;
|
|
|
|
} else {
|
2022-04-20 05:15:38 +00:00
|
|
|
result = to.to_path_buf();
|
|
|
|
break;
|
2021-05-16 18:06:44 +00:00
|
|
|
}
|
|
|
|
}
|
2022-04-20 05:15:38 +00:00
|
|
|
if result.as_os_str().is_empty() {
|
|
|
|
result = PathBuf::from(".");
|
|
|
|
}
|
|
|
|
result
|
2020-12-12 08:00:42 +00:00
|
|
|
}
|
2022-04-22 04:36:56 +00:00
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
2022-09-07 03:29:39 +00:00
|
|
|
use jujutsu_lib::testutils;
|
|
|
|
|
2022-04-22 04:36:56 +00:00
|
|
|
use super::*;
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn parse_file_path_wc_in_cwd() {
|
2022-09-07 03:29:39 +00:00
|
|
|
let temp_dir = testutils::new_temp_dir();
|
2022-04-22 04:36:56 +00:00
|
|
|
let cwd_path = temp_dir.path().join("repo");
|
|
|
|
let wc_path = cwd_path.clone();
|
2022-10-19 00:33:13 +00:00
|
|
|
let unused_stdout = Box::new(Vec::new());
|
|
|
|
let unused_stderr = Box::new(Vec::new());
|
2022-04-22 04:36:56 +00:00
|
|
|
let ui = Ui::new(
|
|
|
|
cwd_path,
|
|
|
|
unused_stdout,
|
|
|
|
unused_stderr,
|
|
|
|
false,
|
|
|
|
UserSettings::default(),
|
|
|
|
);
|
|
|
|
|
|
|
|
assert_eq!(ui.parse_file_path(&wc_path, ""), Ok(RepoPath::root()));
|
|
|
|
assert_eq!(ui.parse_file_path(&wc_path, "."), Ok(RepoPath::root()));
|
|
|
|
assert_eq!(
|
|
|
|
ui.parse_file_path(&wc_path, "file"),
|
|
|
|
Ok(RepoPath::from_internal_string("file"))
|
|
|
|
);
|
|
|
|
// Both slash and the platform's separator are allowed
|
|
|
|
assert_eq!(
|
|
|
|
ui.parse_file_path(&wc_path, &format!("dir{}file", std::path::MAIN_SEPARATOR)),
|
|
|
|
Ok(RepoPath::from_internal_string("dir/file"))
|
|
|
|
);
|
|
|
|
assert_eq!(
|
|
|
|
ui.parse_file_path(&wc_path, "dir/file"),
|
|
|
|
Ok(RepoPath::from_internal_string("dir/file"))
|
|
|
|
);
|
|
|
|
assert_eq!(
|
|
|
|
ui.parse_file_path(&wc_path, ".."),
|
|
|
|
Err(FilePathParseError::InputNotInRepo("..".to_string()))
|
|
|
|
);
|
|
|
|
// TODO: handle these cases:
|
|
|
|
// assert_eq!(ui.parse_file_path(&cwd_path, "../repo"),
|
|
|
|
// Ok(RepoPath::root())); assert_eq!(ui.parse_file_path(&cwd_path,
|
|
|
|
// "../repo/file"), Ok(RepoPath::from_internal_string("file")));
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn parse_file_path_wc_in_cwd_parent() {
|
2022-09-07 03:29:39 +00:00
|
|
|
let temp_dir = testutils::new_temp_dir();
|
2022-04-22 04:36:56 +00:00
|
|
|
let cwd_path = temp_dir.path().join("dir");
|
|
|
|
let wc_path = cwd_path.parent().unwrap().to_path_buf();
|
2022-10-19 00:33:13 +00:00
|
|
|
let unused_stdout = Box::new(Vec::new());
|
|
|
|
let unused_stderr = Box::new(Vec::new());
|
2022-04-22 04:36:56 +00:00
|
|
|
let ui = Ui::new(
|
|
|
|
cwd_path,
|
|
|
|
unused_stdout,
|
|
|
|
unused_stderr,
|
|
|
|
false,
|
|
|
|
UserSettings::default(),
|
|
|
|
);
|
|
|
|
|
|
|
|
assert_eq!(
|
|
|
|
ui.parse_file_path(&wc_path, ""),
|
|
|
|
Ok(RepoPath::from_internal_string("dir"))
|
|
|
|
);
|
|
|
|
assert_eq!(
|
|
|
|
ui.parse_file_path(&wc_path, "."),
|
|
|
|
Ok(RepoPath::from_internal_string("dir"))
|
|
|
|
);
|
|
|
|
assert_eq!(
|
|
|
|
ui.parse_file_path(&wc_path, "file"),
|
|
|
|
Ok(RepoPath::from_internal_string("dir/file"))
|
|
|
|
);
|
|
|
|
assert_eq!(
|
|
|
|
ui.parse_file_path(&wc_path, "subdir/file"),
|
|
|
|
Ok(RepoPath::from_internal_string("dir/subdir/file"))
|
|
|
|
);
|
|
|
|
assert_eq!(ui.parse_file_path(&wc_path, ".."), Ok(RepoPath::root()));
|
|
|
|
assert_eq!(
|
|
|
|
ui.parse_file_path(&wc_path, "../.."),
|
|
|
|
Err(FilePathParseError::InputNotInRepo("../..".to_string()))
|
|
|
|
);
|
|
|
|
assert_eq!(
|
|
|
|
ui.parse_file_path(&wc_path, "../other-dir/file"),
|
|
|
|
Ok(RepoPath::from_internal_string("other-dir/file"))
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn parse_file_path_wc_in_cwd_child() {
|
2022-09-07 03:29:39 +00:00
|
|
|
let temp_dir = testutils::new_temp_dir();
|
2022-04-22 04:36:56 +00:00
|
|
|
let cwd_path = temp_dir.path().join("cwd");
|
|
|
|
let wc_path = cwd_path.join("repo");
|
2022-10-19 00:33:13 +00:00
|
|
|
let unused_stdout = Box::new(Vec::new());
|
|
|
|
let unused_stderr = Box::new(Vec::new());
|
2022-04-22 04:36:56 +00:00
|
|
|
let ui = Ui::new(
|
|
|
|
cwd_path,
|
|
|
|
unused_stdout,
|
|
|
|
unused_stderr,
|
|
|
|
false,
|
|
|
|
UserSettings::default(),
|
|
|
|
);
|
|
|
|
|
|
|
|
assert_eq!(
|
|
|
|
ui.parse_file_path(&wc_path, ""),
|
|
|
|
Err(FilePathParseError::InputNotInRepo("".to_string()))
|
|
|
|
);
|
|
|
|
assert_eq!(
|
|
|
|
ui.parse_file_path(&wc_path, "not-repo"),
|
|
|
|
Err(FilePathParseError::InputNotInRepo("not-repo".to_string()))
|
|
|
|
);
|
|
|
|
assert_eq!(ui.parse_file_path(&wc_path, "repo"), Ok(RepoPath::root()));
|
|
|
|
assert_eq!(
|
|
|
|
ui.parse_file_path(&wc_path, "repo/file"),
|
|
|
|
Ok(RepoPath::from_internal_string("file"))
|
|
|
|
);
|
|
|
|
assert_eq!(
|
|
|
|
ui.parse_file_path(&wc_path, "repo/dir/file"),
|
|
|
|
Ok(RepoPath::from_internal_string("dir/file"))
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|