2013-04-27 18:57:15 -07:00
|
|
|
// 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 <LICENSE-APACHE or
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
|
|
|
|
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
|
|
|
|
// option. This file may not be copied, modified, or distributed
|
|
|
|
// except according to those terms.
|
2013-08-28 01:33:48 -07:00
|
|
|
|
|
|
|
use fmt;
|
2013-09-15 07:27:32 +02:00
|
|
|
use from_str::from_str;
|
2013-08-28 01:33:48 -07:00
|
|
|
use libc::{uintptr_t, exit};
|
2013-09-05 16:15:45 -04:00
|
|
|
use option::{Some, None, Option};
|
2013-08-28 01:33:48 -07:00
|
|
|
use rt;
|
2013-08-30 16:42:49 +02:00
|
|
|
use rt::util::dumb_println;
|
2013-09-27 22:43:41 +02:00
|
|
|
use rt::crate_map::{ModEntry, CrateMap, iter_crate_map, get_crate_map};
|
2013-08-04 21:54:59 -07:00
|
|
|
use str::StrSlice;
|
2013-08-30 16:42:49 +02:00
|
|
|
use str::raw::from_c_str;
|
|
|
|
use u32;
|
|
|
|
use vec::ImmutableVector;
|
2013-09-27 22:43:41 +02:00
|
|
|
#[cfg(test)] use cast::transmute;
|
2013-08-30 16:42:49 +02:00
|
|
|
|
|
|
|
struct LogDirective {
|
2013-09-05 21:18:55 -04:00
|
|
|
name: Option<~str>,
|
2013-08-30 16:42:49 +02:00
|
|
|
level: u32
|
|
|
|
}
|
|
|
|
|
|
|
|
static MAX_LOG_LEVEL: u32 = 255;
|
|
|
|
static DEFAULT_LOG_LEVEL: u32 = 1;
|
2013-09-05 16:15:45 -04:00
|
|
|
static log_level_names : &'static[&'static str] = &'static["error", "warn", "info", "debug"];
|
|
|
|
|
|
|
|
/// Parse an individual log level that is either a number or a symbolic log level
|
|
|
|
fn parse_log_level(level: &str) -> Option<u32> {
|
2013-09-15 07:27:32 +02:00
|
|
|
let num = from_str::<u32>(level);
|
2013-09-05 16:15:45 -04:00
|
|
|
let mut log_level;
|
2013-09-05 21:18:55 -04:00
|
|
|
match num {
|
2013-09-05 16:15:45 -04:00
|
|
|
Some(num) => {
|
|
|
|
if num < MAX_LOG_LEVEL {
|
|
|
|
log_level = Some(num);
|
|
|
|
} else {
|
|
|
|
log_level = Some(MAX_LOG_LEVEL);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
_ => {
|
|
|
|
let position = log_level_names.iter().position(|&name| name == level);
|
2013-09-05 21:18:55 -04:00
|
|
|
match position {
|
2013-09-05 16:15:45 -04:00
|
|
|
Some(position) => {
|
2013-09-05 21:18:55 -04:00
|
|
|
log_level = Some(u32::min(MAX_LOG_LEVEL, (position + 1) as u32))
|
2013-09-05 16:15:45 -04:00
|
|
|
},
|
|
|
|
_ => {
|
|
|
|
log_level = None;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
log_level
|
|
|
|
}
|
|
|
|
|
2013-08-30 16:42:49 +02:00
|
|
|
/// Parse a logging specification string (e.g: "crate1,crate2::mod3,crate3::x=1")
|
|
|
|
/// and return a vector with log directives.
|
2013-09-05 16:15:45 -04:00
|
|
|
/// Valid log levels are 0-255, with the most likely ones being 1-4 (defined in std::).
|
|
|
|
/// Also supports string log levels of error, warn, info, and debug
|
2013-08-30 16:42:49 +02:00
|
|
|
fn parse_logging_spec(spec: ~str) -> ~[LogDirective]{
|
|
|
|
let mut dirs = ~[];
|
|
|
|
for s in spec.split_iter(',') {
|
|
|
|
let parts: ~[&str] = s.split_iter('=').collect();
|
2013-09-05 16:15:45 -04:00
|
|
|
let mut log_level;
|
2013-09-05 21:18:55 -04:00
|
|
|
let mut name = Some(parts[0].to_owned());
|
2013-08-30 16:42:49 +02:00
|
|
|
match parts.len() {
|
2013-09-05 21:18:55 -04:00
|
|
|
1 => {
|
|
|
|
//if the single argument is a log-level string or number,
|
|
|
|
//treat that as a global fallback
|
|
|
|
let possible_log_level = parse_log_level(parts[0]);
|
|
|
|
match possible_log_level {
|
|
|
|
Some(num) => {
|
|
|
|
name = None;
|
|
|
|
log_level = num;
|
|
|
|
},
|
|
|
|
_ => {
|
|
|
|
log_level = MAX_LOG_LEVEL
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2013-08-30 16:42:49 +02:00
|
|
|
2 => {
|
2013-09-05 16:15:45 -04:00
|
|
|
let possible_log_level = parse_log_level(parts[1]);
|
|
|
|
match possible_log_level {
|
2013-08-30 16:42:49 +02:00
|
|
|
Some(num) => {
|
2013-09-05 16:15:45 -04:00
|
|
|
log_level = num;
|
|
|
|
},
|
2013-08-30 16:42:49 +02:00
|
|
|
_ => {
|
2013-09-27 17:02:31 -07:00
|
|
|
dumb_println(format!("warning: invalid logging spec \
|
|
|
|
'{}', ignoring it", parts[1]));
|
2013-10-01 14:31:03 -07:00
|
|
|
continue;
|
2013-08-30 16:42:49 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
_ => {
|
2013-09-27 17:02:31 -07:00
|
|
|
dumb_println(format!("warning: invalid logging spec '{}',\
|
|
|
|
ignoring it", s));
|
2013-10-01 14:31:03 -07:00
|
|
|
continue;
|
2013-08-30 16:42:49 +02:00
|
|
|
}
|
|
|
|
}
|
2013-09-05 21:18:55 -04:00
|
|
|
let dir = LogDirective {name: name, level: log_level};
|
2013-08-30 16:42:49 +02:00
|
|
|
dirs.push(dir);
|
|
|
|
}
|
|
|
|
return dirs;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Set the log level of an entry in the crate map depending on the vector
|
|
|
|
/// of log directives
|
2013-09-27 22:43:41 +02:00
|
|
|
fn update_entry(dirs: &[LogDirective], entry: &ModEntry) -> u32 {
|
2013-08-30 16:42:49 +02:00
|
|
|
let mut new_lvl: u32 = DEFAULT_LOG_LEVEL;
|
2013-09-05 21:18:55 -04:00
|
|
|
let mut longest_match = -1i;
|
2013-08-30 16:42:49 +02:00
|
|
|
unsafe {
|
|
|
|
for dir in dirs.iter() {
|
2013-09-05 21:18:55 -04:00
|
|
|
match dir.name {
|
|
|
|
None => {
|
|
|
|
if longest_match == -1 {
|
|
|
|
longest_match = 0;
|
|
|
|
new_lvl = dir.level;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Some(ref dir_name) => {
|
2013-09-15 21:41:02 +02:00
|
|
|
let name = from_c_str(entry.name);
|
2013-09-05 21:18:55 -04:00
|
|
|
let len = dir_name.len() as int;
|
|
|
|
if name.starts_with(*dir_name) &&
|
|
|
|
len >= longest_match {
|
|
|
|
longest_match = len;
|
|
|
|
new_lvl = dir.level;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
2013-08-30 16:42:49 +02:00
|
|
|
}
|
2013-09-15 21:41:02 +02:00
|
|
|
*entry.log_level = new_lvl;
|
2013-08-30 16:42:49 +02:00
|
|
|
}
|
2013-09-05 21:18:55 -04:00
|
|
|
if longest_match >= 0 { return 1; } else { return 0; }
|
2013-08-30 16:42:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
#[fixed_stack_segment] #[inline(never)]
|
|
|
|
/// Set log level for every entry in crate_map according to the sepecification
|
|
|
|
/// in settings
|
2013-09-27 22:43:41 +02:00
|
|
|
fn update_log_settings(crate_map: &CrateMap, settings: ~str) {
|
2013-08-30 16:42:49 +02:00
|
|
|
let mut dirs = ~[];
|
|
|
|
if settings.len() > 0 {
|
|
|
|
if settings == ~"::help" || settings == ~"?" {
|
|
|
|
dumb_println("\nCrate log map:\n");
|
2013-09-10 00:04:29 +02:00
|
|
|
unsafe {
|
2013-09-15 21:41:02 +02:00
|
|
|
do iter_crate_map(crate_map) |entry| {
|
|
|
|
dumb_println(" "+from_c_str(entry.name));
|
2013-08-30 16:42:49 +02:00
|
|
|
}
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
dirs = parse_logging_spec(settings);
|
|
|
|
}
|
|
|
|
|
|
|
|
let mut n_matches: u32 = 0;
|
2013-09-15 21:41:02 +02:00
|
|
|
do iter_crate_map(crate_map) |entry| {
|
|
|
|
let m = update_entry(dirs, entry);
|
|
|
|
n_matches += m;
|
2013-08-30 16:42:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if n_matches < (dirs.len() as u32) {
|
2013-09-27 17:02:31 -07:00
|
|
|
dumb_println(format!("warning: got {} RUST_LOG specs but only matched\n\
|
|
|
|
{} of them. You may have mistyped a RUST_LOG \
|
|
|
|
spec. \n\
|
|
|
|
Use RUST_LOG=::help to see the list of crates \
|
|
|
|
and modules.\n",
|
|
|
|
dirs.len(), n_matches));
|
2013-08-30 16:42:49 +02:00
|
|
|
}
|
|
|
|
}
|
2013-04-27 18:57:15 -07:00
|
|
|
|
|
|
|
pub trait Logger {
|
2013-08-28 01:33:48 -07:00
|
|
|
fn log(&mut self, args: &fmt::Arguments);
|
2013-04-27 18:57:15 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
pub struct StdErrLogger;
|
|
|
|
|
|
|
|
impl Logger for StdErrLogger {
|
2013-08-28 01:33:48 -07:00
|
|
|
fn log(&mut self, args: &fmt::Arguments) {
|
|
|
|
if should_log_console() {
|
2013-09-29 12:39:49 -07:00
|
|
|
fmt::writeln(self as &mut rt::io::Writer, args);
|
2013-06-21 16:52:07 -07:00
|
|
|
}
|
2013-08-28 01:33:48 -07:00
|
|
|
}
|
|
|
|
}
|
2013-06-21 16:52:07 -07:00
|
|
|
|
2013-08-28 01:33:48 -07:00
|
|
|
impl rt::io::Writer for StdErrLogger {
|
|
|
|
fn write(&mut self, buf: &[u8]) {
|
|
|
|
// Nothing like swapping between I/O implementations! In theory this
|
|
|
|
// could use the libuv bindings for writing to file descriptors, but
|
|
|
|
// that may not necessarily be desirable because logging should work
|
|
|
|
// outside of the uv loop. (modify with caution)
|
|
|
|
use io::Writer;
|
|
|
|
let dbg = ::libc::STDERR_FILENO as ::io::fd_t;
|
|
|
|
dbg.write(buf);
|
2013-04-27 18:57:15 -07:00
|
|
|
}
|
2013-08-28 01:33:48 -07:00
|
|
|
|
|
|
|
fn flush(&mut self) {}
|
2013-05-08 16:53:40 -07:00
|
|
|
}
|
2013-08-28 01:33:48 -07:00
|
|
|
|
2013-05-08 16:53:40 -07:00
|
|
|
/// Configure logging by traversing the crate map and setting the
|
|
|
|
/// per-module global logging flags based on the logging spec
|
2013-09-18 04:58:52 -04:00
|
|
|
pub fn init() {
|
|
|
|
use os;
|
|
|
|
|
2013-09-27 22:43:41 +02:00
|
|
|
let crate_map = get_crate_map();
|
2013-09-18 04:58:52 -04:00
|
|
|
|
2013-05-08 16:53:40 -07:00
|
|
|
let log_spec = os::getenv("RUST_LOG");
|
|
|
|
match log_spec {
|
|
|
|
Some(spec) => {
|
2013-08-30 16:42:49 +02:00
|
|
|
update_log_settings(crate_map, spec);
|
2013-05-08 16:53:40 -07:00
|
|
|
}
|
|
|
|
None => {
|
2013-08-30 16:42:49 +02:00
|
|
|
update_log_settings(crate_map, ~"");
|
2013-05-08 16:53:40 -07:00
|
|
|
}
|
|
|
|
}
|
2013-06-21 16:52:07 -07:00
|
|
|
}
|
2013-05-08 16:53:40 -07:00
|
|
|
|
2013-08-14 21:41:40 -04:00
|
|
|
#[fixed_stack_segment] #[inline(never)]
|
2013-06-21 16:52:07 -07:00
|
|
|
pub fn console_on() { unsafe { rust_log_console_on() } }
|
2013-08-14 21:41:40 -04:00
|
|
|
|
|
|
|
#[fixed_stack_segment] #[inline(never)]
|
2013-06-21 16:52:07 -07:00
|
|
|
pub fn console_off() { unsafe { rust_log_console_off() } }
|
2013-08-14 21:41:40 -04:00
|
|
|
|
|
|
|
#[fixed_stack_segment] #[inline(never)]
|
2013-06-21 16:52:07 -07:00
|
|
|
fn should_log_console() -> bool { unsafe { rust_should_log_console() != 0 } }
|
|
|
|
|
|
|
|
extern {
|
|
|
|
fn rust_log_console_on();
|
|
|
|
fn rust_log_console_off();
|
2013-08-30 16:42:49 +02:00
|
|
|
fn rust_should_log_console() -> uintptr_t;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Tests for parse_logging_spec()
|
|
|
|
#[test]
|
|
|
|
fn parse_logging_spec_valid() {
|
2013-09-05 21:18:55 -04:00
|
|
|
let dirs = parse_logging_spec(~"crate1::mod1=1,crate1::mod2,crate2=4");
|
2013-08-30 16:42:49 +02:00
|
|
|
assert_eq!(dirs.len(), 3);
|
2013-09-05 21:18:55 -04:00
|
|
|
assert!(dirs[0].name == Some(~"crate1::mod1"));
|
2013-08-30 16:42:49 +02:00
|
|
|
assert_eq!(dirs[0].level, 1);
|
|
|
|
|
2013-09-05 21:18:55 -04:00
|
|
|
assert!(dirs[1].name == Some(~"crate1::mod2"));
|
2013-08-30 16:42:49 +02:00
|
|
|
assert_eq!(dirs[1].level, MAX_LOG_LEVEL);
|
|
|
|
|
2013-09-05 21:18:55 -04:00
|
|
|
assert!(dirs[2].name == Some(~"crate2"));
|
2013-08-30 16:42:49 +02:00
|
|
|
assert_eq!(dirs[2].level, 4);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn parse_logging_spec_invalid_crate() {
|
|
|
|
// test parse_logging_spec with multiple = in specification
|
2013-09-05 21:18:55 -04:00
|
|
|
let dirs = parse_logging_spec(~"crate1::mod1=1=2,crate2=4");
|
2013-08-30 16:42:49 +02:00
|
|
|
assert_eq!(dirs.len(), 1);
|
2013-09-05 21:18:55 -04:00
|
|
|
assert!(dirs[0].name == Some(~"crate2"));
|
2013-08-30 16:42:49 +02:00
|
|
|
assert_eq!(dirs[0].level, 4);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn parse_logging_spec_invalid_log_level() {
|
|
|
|
// test parse_logging_spec with 'noNumber' as log level
|
2013-09-05 21:18:55 -04:00
|
|
|
let dirs = parse_logging_spec(~"crate1::mod1=noNumber,crate2=4");
|
2013-08-30 16:42:49 +02:00
|
|
|
assert_eq!(dirs.len(), 1);
|
2013-09-05 21:18:55 -04:00
|
|
|
assert!(dirs[0].name == Some(~"crate2"));
|
2013-08-30 16:42:49 +02:00
|
|
|
assert_eq!(dirs[0].level, 4);
|
|
|
|
}
|
|
|
|
|
2013-09-05 16:15:45 -04:00
|
|
|
#[test]
|
|
|
|
fn parse_logging_spec_string_log_level() {
|
|
|
|
// test parse_logging_spec with 'warn' as log level
|
2013-09-05 21:18:55 -04:00
|
|
|
let dirs = parse_logging_spec(~"crate1::mod1=wrong,crate2=warn");
|
2013-09-05 16:15:45 -04:00
|
|
|
assert_eq!(dirs.len(), 1);
|
2013-09-05 21:18:55 -04:00
|
|
|
assert!(dirs[0].name == Some(~"crate2"));
|
2013-09-05 16:15:45 -04:00
|
|
|
assert_eq!(dirs[0].level, 2);
|
|
|
|
}
|
|
|
|
|
2013-09-05 21:18:55 -04:00
|
|
|
#[test]
|
|
|
|
fn parse_logging_spec_global() {
|
|
|
|
// test parse_logging_spec with no crate
|
|
|
|
let dirs = parse_logging_spec(~"warn,crate2=4");
|
|
|
|
assert_eq!(dirs.len(), 2);
|
|
|
|
assert!(dirs[0].name == None);
|
|
|
|
assert_eq!(dirs[0].level, 2);
|
|
|
|
assert!(dirs[1].name == Some(~"crate2"));
|
|
|
|
assert_eq!(dirs[1].level, 4);
|
|
|
|
}
|
|
|
|
|
2013-08-30 16:42:49 +02:00
|
|
|
// Tests for update_entry
|
|
|
|
#[test]
|
|
|
|
fn update_entry_match_full_path() {
|
|
|
|
use c_str::ToCStr;
|
2013-09-05 21:18:55 -04:00
|
|
|
let dirs = ~[LogDirective {name: Some(~"crate1::mod1"), level: 2 },
|
|
|
|
LogDirective {name: Some(~"crate2"), level: 3}];
|
|
|
|
let level = &mut 0;
|
2013-08-30 16:42:49 +02:00
|
|
|
unsafe {
|
2013-09-18 10:54:43 -07:00
|
|
|
do "crate1::mod1".with_c_str |ptr| {
|
2013-09-05 21:18:55 -04:00
|
|
|
let entry= &ModEntry {name: ptr, log_level: level};
|
2013-08-30 16:42:49 +02:00
|
|
|
let m = update_entry(dirs, transmute(entry));
|
|
|
|
assert!(*entry.log_level == 2);
|
|
|
|
assert!(m == 1);
|
|
|
|
}
|
|
|
|
}
|
2013-05-08 16:53:40 -07:00
|
|
|
}
|
2013-06-21 16:52:07 -07:00
|
|
|
|
2013-08-30 16:42:49 +02:00
|
|
|
#[test]
|
|
|
|
fn update_entry_no_match() {
|
|
|
|
use c_str::ToCStr;
|
2013-09-05 21:18:55 -04:00
|
|
|
let dirs = ~[LogDirective {name: Some(~"crate1::mod1"), level: 2 },
|
|
|
|
LogDirective {name: Some(~"crate2"), level: 3}];
|
|
|
|
let level = &mut 0;
|
2013-08-30 16:42:49 +02:00
|
|
|
unsafe {
|
2013-09-18 10:54:43 -07:00
|
|
|
do "crate3::mod1".with_c_str |ptr| {
|
2013-09-05 21:18:55 -04:00
|
|
|
let entry= &ModEntry {name: ptr, log_level: level};
|
2013-08-30 16:42:49 +02:00
|
|
|
let m = update_entry(dirs, transmute(entry));
|
|
|
|
assert!(*entry.log_level == DEFAULT_LOG_LEVEL);
|
|
|
|
assert!(m == 0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn update_entry_match_beginning() {
|
|
|
|
use c_str::ToCStr;
|
2013-09-05 21:18:55 -04:00
|
|
|
let dirs = ~[LogDirective {name: Some(~"crate1::mod1"), level: 2 },
|
|
|
|
LogDirective {name: Some(~"crate2"), level: 3}];
|
|
|
|
let level = &mut 0;
|
2013-08-30 16:42:49 +02:00
|
|
|
unsafe {
|
2013-09-18 10:54:43 -07:00
|
|
|
do "crate2::mod1".with_c_str |ptr| {
|
2013-09-05 21:18:55 -04:00
|
|
|
let entry= &ModEntry {name: ptr, log_level: level};
|
2013-08-30 16:42:49 +02:00
|
|
|
let m = update_entry(dirs, transmute(entry));
|
|
|
|
assert!(*entry.log_level == 3);
|
|
|
|
assert!(m == 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn update_entry_match_beginning_longest_match() {
|
|
|
|
use c_str::ToCStr;
|
2013-09-05 21:18:55 -04:00
|
|
|
let dirs = ~[LogDirective {name: Some(~"crate1::mod1"), level: 2 },
|
|
|
|
LogDirective {name: Some(~"crate2"), level: 3},
|
|
|
|
LogDirective {name: Some(~"crate2::mod"), level: 4}];
|
|
|
|
let level = &mut 0;
|
2013-08-30 16:42:49 +02:00
|
|
|
unsafe {
|
2013-09-18 10:54:43 -07:00
|
|
|
do "crate2::mod1".with_c_str |ptr| {
|
2013-09-05 21:18:55 -04:00
|
|
|
let entry = &ModEntry {name: ptr, log_level: level};
|
2013-08-30 16:42:49 +02:00
|
|
|
let m = update_entry(dirs, transmute(entry));
|
|
|
|
assert!(*entry.log_level == 4);
|
|
|
|
assert!(m == 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2013-09-05 21:18:55 -04:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn update_entry_match_default() {
|
|
|
|
use c_str::ToCStr;
|
|
|
|
let dirs = ~[LogDirective {name: Some(~"crate1::mod1"), level: 2 },
|
|
|
|
LogDirective {name: None, level: 3}
|
|
|
|
];
|
|
|
|
let level = &mut 0;
|
|
|
|
unsafe {
|
2013-09-18 10:54:43 -07:00
|
|
|
do "crate1::mod1".with_c_str |ptr| {
|
2013-09-05 21:18:55 -04:00
|
|
|
let entry= &ModEntry {name: ptr, log_level: level};
|
|
|
|
let m = update_entry(dirs, transmute(entry));
|
|
|
|
assert!(*entry.log_level == 2);
|
|
|
|
assert!(m == 1);
|
|
|
|
}
|
2013-09-18 10:54:43 -07:00
|
|
|
do "crate2::mod2".with_c_str |ptr| {
|
2013-09-05 21:18:55 -04:00
|
|
|
let entry= &ModEntry {name: ptr, log_level: level};
|
|
|
|
let m = update_entry(dirs, transmute(entry));
|
|
|
|
assert!(*entry.log_level == 3);
|
|
|
|
assert!(m == 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|