-
Notifications
You must be signed in to change notification settings - Fork 519
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Building a training set of tags for rust #1760
Comments
Exercise: raindropsCodepub fn raindrops(n: usize) -> String {
let mut output = String::new();
if n % 3 == 0 {
output.push_str("Pling");
}
if n % 5 == 0 {
output.push_str("Plang");
}
if n % 7 == 0 {
output.push_str("Plong");
}
// If our output has no characters, return the int as a string
if output == "" {
output.push_str(&n.to_string());
}
output
} Tags:
|
Exercise: sum-of-multiplesCodepub fn sum_of_multiples(n: u64, m: &[u64]) -> u64 {
(0..n).filter(|&i| {
m.iter().any(|&x| x != 0 && i % x == 0)
}).sum()
} Tags:
|
Exercise: seriesCodepub fn series(digits: &str, len: usize) -> Vec<String> {
digits
.chars()
.collect::<Vec<char>>()
.windows(len)
.map(|w| w.iter().collect::<String>())
.collect()
} Tags:
|
Exercise: diffie-hellmanCodeextern crate rand;
use rand::prelude::*;
pub fn private_key(p: u64) -> u64 {
thread_rng().gen_range(2, p)
}
pub fn public_key(p: u64, g: u64, a: u64) -> u64 {
modular_pow(g, a, p)
}
pub fn secret(p: u64, b_pub: u64, a: u64) -> u64 {
modular_pow(b_pub, a, p)
}
//Based on wikipedia's explanation
fn modular_pow(a: u64, b: u64, m: u64) -> u64 {
if m == 1 { return 0; }
let mut c = 1;
for _x in 0..b {
c = (c * a) % m;
}
c
} Tags:
|
Exercise: clockCode#![feature(euclidean_division)]
use std::fmt;
#[derive(Debug, PartialEq)]
pub struct Clock {
minutes: i32,
}
const MINS_PER_DAY: i32 = 60 * 24;
impl Clock {
pub fn new(hours: i32, minutes: i32) -> Self {
Clock {
minutes: (hours * 60 + minutes).mod_euc(MINS_PER_DAY),
}
}
pub fn add_minutes(self, minutes: i32) -> Self {
Clock::new(0, self.minutes + minutes)
}
}
impl fmt::Display for Clock {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
write!(f, "{:02}:{:02}", self.minutes / 60, self.minutes % 60)
}
} Tags:
|
Exercise: pascals-triangleCode#![feature(iter_unfold)]
use std::iter::successors;
pub struct PascalsTriangle {
row_count: u32,
}
impl PascalsTriangle {
pub fn new(row_count: u32) -> Self {
Self { row_count }
}
pub fn rows(&self) -> Vec<Vec<u32>> {
successors(Some(vec![1]), |row| {
let mut next = row.to_owned();
next.push(0);
for (a, b) in next.iter_mut().skip(1).zip(row).rev() {
*a += b;
}
Some(next)
})
.take(self.row_count as usize)
.collect()
}
} Tags:
|
Exercise: all-your-baseCode#[derive(Debug, PartialEq)]
pub enum Error {
InvalidInputBase,
InvalidOutputBase,
InvalidDigit(u32),
}
///
/// Convert a number between two bases.
///
/// A number is any slice of digits.
/// A digit is any unsigned integer (e.g. u8, u16, u32, u64, or usize).
/// Bases are specified as unsigned integers.
///
/// Return an `Err(.)` if the conversion is impossible.
/// The tests do not test for specific values inside the `Err(.)`.
///
///
/// You are allowed to change the function signature as long as all test still pass.
///
///
/// Example:
/// Input
/// number: &[4, 2]
/// from_base: 10
/// to_base: 2
/// Result
/// Ok(vec![1, 0, 1, 0, 1, 0])
///
/// The example corresponds to converting the number 42 from decimal
/// which is equivalent to 101010 in binary.
///
///
/// Notes:
/// * The empty slice ( "[]" ) is equal to the number 0.
/// * Never output leading 0 digits. However, your function must be able to
/// process input with leading 0 digits.
///
pub fn convert(number: &[u32], from_base: u32, to_base: u32) -> Result<Vec<u32>, Error> {
if from_base < 2 { return Err(Error::InvalidInputBase) }
if to_base < 2 { return Err(Error::InvalidOutputBase) }
let base10_value = convert_to_base10(number, from_base)?;
Ok(convert_int_to_base(base10_value, to_base))
}
fn convert_to_base10(number: &[u32], from_base: u32) -> Result<u32, Error> {
let mut result = 0;
for (i, &digit) in number.iter().rev().enumerate() {
if digit >= from_base { return Err(Error::InvalidDigit(digit))}
result += from_base.pow(i as u32) * digit;
}
Ok(result)
}
fn convert_int_to_base(mut value: u32, to_base: u32) -> Vec<u32> {
let mut result = vec![];
while value != 0 {
result.push(value % to_base);
value /= to_base;
}
result.into_iter().rev().collect()
} Tags:
|
Exercise: robot-simulatorCode#[derive(PartialEq, Debug)]
pub enum Direction {
North,
East,
South,
West,
}
pub struct Robot {
x: isize,
y: isize,
d: Direction,
}
impl Robot {
pub fn new(x: isize, y: isize, d: Direction) -> Self {
Robot { x, y, d }
}
pub fn turn_right(self) -> Self {
Robot {
x: self.x,
y: self.y,
d: match self.d {
Direction::North => Direction::East,
Direction::East => Direction::South,
Direction::South => Direction::West,
Direction::West => Direction::North,
},
}
}
pub fn turn_left(self) -> Self {
Robot {
x: self.x,
y: self.y,
d: match self.d {
Direction::North => Direction::West,
Direction::West => Direction::South,
Direction::South => Direction::East,
Direction::East => Direction::North,
},
}
}
pub fn advance(self) -> Self {
Robot {
x: match self.d {
Direction::East => self.x + 1,
Direction::West => self.x - 1,
_ => self.x,
},
y: match self.d {
Direction::North => self.y + 1,
Direction::South => self.y - 1,
_ => self.y,
},
d: self.d,
}
}
pub fn instructions(self, instructions: &str) -> Self {
let mut robot = self;
for command in instructions.chars() {
match command {
'L' => robot = robot.turn_left(),
'R' => robot = robot.turn_right(),
'A' => robot = robot.advance(),
_ => panic!("Illegal command"),
}
}
robot
}
pub fn position(&self) -> (isize, isize) {
(self.x, self.y)
}
pub fn direction(&self) -> &Direction {
&self.d
}
} Tags:
|
Exercise: matching-bracketsCodeuse std::vec::Vec;
pub fn brackets_are_balanced(string: &str) -> bool {
let mut stack: Vec<char> = Vec::new();
for (single_char) in string.chars() {
match single_char {
'(' => {
stack.push(single_char);
continue;
}
'[' => {
stack.push(single_char);
continue;
}
'{' => {
stack.push(single_char);
continue;
}
'}' if stack.last() == Some(&'{') => {
stack.pop();
continue;
}
']' if stack.last() == Some(&'[') => {
stack.pop();
continue;
}
')' if stack.last() == Some(&'(') => {
stack.pop();
continue;
}
'}' if stack.last() != Some(&'{') => {
return false;
}
']' if stack.last() != Some(&'[') => {
return false;
}
')' if stack.last() != Some(&'(') => {
return false;
}
_ => {}
}
}
if stack.is_empty() {
true
} else {
false
}
} Tags:
|
Exercise: bowlingCode#[derive(Debug, PartialEq)]
pub enum Error {
NotEnoughPinsLeft,
GameComplete,
}
pub struct BowlingGame {
pub frames: Vec<Frame>,
pub current_frame: Frame,
}
#[derive(Copy, Clone)]
pub struct Frame {
pub is_spare: bool,
pub is_strike: bool,
pub shots_done: u16,
pub first: u16,
pub second:u16,
pub third:u16
}
impl BowlingGame {
pub fn new() -> Self {
BowlingGame{frames:Vec::new(),
current_frame:Frame{
is_strike:false, is_spare: false,
first:0, second:0, third:0, shots_done:0}}
}
pub fn roll(&mut self, pins: u16) -> Result<(), Error> {
if self.frames.len() == 10 {
return Err(Error::GameComplete);
}
//X frame
if self.frames.len() == 9 {
if self.current_frame.shots_done == 0 {
if pins > 10 {
return Err(Error::NotEnoughPinsLeft);
}
if pins == 10 {
self.current_frame.is_strike = true;
}
self.current_frame.first = pins;
self.current_frame.shots_done = 1;
return Ok(());
}
if self.current_frame.shots_done == 1 {
if pins > 10 {
return Err(Error::NotEnoughPinsLeft);
}
if !self.current_frame.is_strike && (self.current_frame.first + pins) > 10 {
return Err(Error::NotEnoughPinsLeft);
}
if pins == 10 {
self.current_frame.is_strike = true;
}
if pins + self.current_frame.first == 10 {
self.current_frame.is_spare = true;
}
self.current_frame.second = pins;
self.current_frame.shots_done = 2;
if !self.current_frame.is_spare && !self.current_frame.is_strike {
self.finish_frame();
}
return Ok(());
}
if self.current_frame.shots_done == 2 &&
(self.current_frame.is_spare || self.current_frame.is_strike) {
if pins > 10 {
return Err(Error::NotEnoughPinsLeft);
}
if self.current_frame.second!=10 && !self.current_frame.is_spare {
if pins == 10 || self.current_frame.second + pins > 10{
return Err(Error::NotEnoughPinsLeft);
}
}
self.current_frame.third = pins;
self.finish_frame();
return Ok(());
}
}
if self.current_frame.shots_done == 0 {
if pins > 10 {
return Err(Error::NotEnoughPinsLeft);
}
//Strike
if pins == 10 {
self.current_frame.is_strike = true;
self.current_frame.first = 10;
self.finish_frame();
} else {
//normal
self.current_frame.shots_done = 1;
self.current_frame.first = pins;
}
return Ok(());
}
if self.current_frame.shots_done == 1 {
if pins + self.current_frame.first > 10 {
return Err(Error::NotEnoughPinsLeft);
}
//Spare
if pins + self.current_frame.first == 10 {
self.current_frame.is_spare = true;
}
self.current_frame.shots_done = 2;
self.current_frame.second = pins;
self.finish_frame();
}
Ok(())
}
fn finish_frame(&mut self) {
self.frames.push(self.current_frame);
self.current_frame = Frame{
is_strike:false, is_spare: false,
first:0, second:0, third:0, shots_done:0};
}
pub fn score(&self) -> Option<u16> {
if self.frames.len() == 10 {
let mut total = 0;
for i in 0..10 {
let frame = self.frames[i];
if i < 9 {
total += frame.first + frame.second;
if frame.is_spare {
total += self.frames[i+1].first;
}
if frame.is_strike {
if self.frames[i+1].is_strike && i < 8 {
total += self.frames[i+1].first + self.frames[i+2].first;
} else {
total += self.frames[i+1].first + self.frames[i+1].second;
}
}
}
else {
total += frame.first + frame.second + frame.third;
}
}
Some(total)
}
else {
None
}
}
} Tags:
|
Exercise: phone-numberCodepub fn number(input: &str) -> Option<String> {
input
.chars()
.filter(|c| c.is_ascii_digit())
.enumerate()
.filter(|&(i, c)| !(i == 0 && c == '1'))
.enumerate()
.map(|(i, (_, c))| match c {
'0'...'1' if [0, 3].contains(&i) => None,
_ => Some(c),
})
.collect::<Option<String>>()
.filter(|s| s.len() == 10)
} Tags:
|
Exercise: tournamentCodeuse std::cmp::Ordering;
use std::collections::HashMap;
use std::fmt;
use std::iter::FromIterator;
#[derive(Debug)]
enum MatchResult {
Win,
Loose,
Draw,
}
#[derive(Debug)]
struct Stats {
win: u32,
loose: u32,
draw: u32,
}
struct Match {
team_a: String,
team_b: String,
result: MatchResult,
}
impl Stats {
fn points(&self) -> u32 {
3 * self.win + self.draw
}
fn played(&self) -> u32 {
self.win + self.loose + self.draw
}
fn new() -> Self {
Stats {
win: 0,
loose: 0,
draw: 0,
}
}
}
impl fmt::Display for Stats {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
write!(
f,
"{} | {} | {} | {} | {}",
self.played(),
self.win,
self.draw,
self.loose,
self.points()
)
}
}
impl<'a> FromIterator<&'a str> for Match {
fn from_iter<I: IntoIterator<Item = &'a str>>(iter: I) -> Self {
let mut iter = iter.into_iter();
let team_a = iter.next().unwrap().to_owned();
let team_b = iter.next().unwrap().to_owned();
let result = match iter.next().unwrap() {
"draw" => MatchResult::Draw,
"win" => MatchResult::Win,
_ => MatchResult::Loose,
};
Match {
team_a,
team_b,
result,
}
}
}
pub fn tally(match_results: &str) -> String {
let mut teams = HashMap::new();
for line in match_results.lines() {
let match_result = line.split(";").collect::<Match>();
match match_result.result {
MatchResult::Win => {
teams.entry(match_result.team_a).or_insert(Stats::new()).win += 1;
teams
.entry(match_result.team_b)
.or_insert(Stats::new())
.loose += 1;
}
MatchResult::Loose => {
teams
.entry(match_result.team_a)
.or_insert(Stats::new())
.loose += 1;
teams.entry(match_result.team_b).or_insert(Stats::new()).win += 1;
}
MatchResult::Draw => {
teams
.entry(match_result.team_a)
.or_insert(Stats::new())
.draw += 1;
teams
.entry(match_result.team_b)
.or_insert(Stats::new())
.draw += 1;
}
};
}
let mut teams_vec = teams.iter().collect::<Vec<_>>();
println!("{:?}", teams_vec);
teams_vec.sort_by(|(team_a, stats_a), (team_b, stats_b)| {
match (team_a.cmp(team_b), stats_b.points().cmp(&stats_a.points())) {
(a, Ordering::Equal) => a,
(_, a) => a,
}
});
println!("{:?}", teams_vec);
teams_vec.iter().fold(
"Team | MP | W | D | L | P".to_string(),
|accum, (team_name, stats)| format!("{}\n{:30} | {}", accum, team_name, stats),
)
} Tags:
|
Exercise: alphameticsCodeuse std::collections::{HashMap, HashSet};
struct AlphaNumMap(Vec<char>);
impl AlphaNumMap {
pub fn new() -> Self {
Self(Vec::with_capacity(10))
}
pub fn push(&mut self, c: char) {
self.0.push(c);
}
pub fn pop(&mut self) {
self.0.pop();
}
pub fn value(&self, c: char) -> u64 {
self.0.iter().position(|&e| e == c).map(|n| n as u64).expect(&format!("Unmmapped {:?}", c))
}
pub fn to_hashmap(&self) -> Option<HashMap<char, u8>> {
if self.0.len() > 0 {
Some(self.0.iter()
.enumerate()
.filter(|(_,&c)| c != ' ')
.map(|(i,&c)| (c, i as u8))
.collect())
} else {
None
}
}
}
struct AlphaNum<'a>(&'a str);
impl<'a> AlphaNum<'a> {
pub fn from_equality(input: &'a str) -> (Vec<Self>,Self) {
let equality = input.split(" == ").collect::<Vec<_>>();
let (left, right) = match equality.len() {
0 | 1 => panic!("Missing ' == '"),
2 => (equality[0], equality[1]),
_ => panic!("Too much ' == '"),
};
(AlphaNum::from_sum(left), AlphaNum::new(right))
}
pub fn from_sum(input: &'a str) -> Vec<Self> {
input.split(" + ")
.map(AlphaNum::new)
.collect()
}
pub fn new(input: &'a str) -> Self {
if input.len() == 0 || input.chars().any(|c| c < 'A' || c > 'Z') {
panic!("Invalid term {:?}", input);
}
AlphaNum(input)
}
pub fn to_u64(&self, map: &AlphaNumMap) -> u64 {
self.0.chars()
.fold(0u64, |acc, c| acc * 10 + map.value(c))
}
pub fn leading(&self) -> char {
self.0.chars().nth(0).expect("Empty term")
}
pub fn len(&self) -> usize {
self.0.len()
}
}
pub fn solve(input: &str) -> Option<HashMap<char, u8>> {
let letters = input.chars()
.filter(|&c| c >= 'A' && c <= 'Z')
.collect::<HashSet<_>>();
if letters.len() > 10 {
return None
}
let (terms, result) = AlphaNum::from_equality(input);
let leadings = terms.iter().chain([&result].into_iter().cloned())
.filter(|n| n.len() > 1)
.map(AlphaNum::leading)
.collect::<HashSet<_>>();
let mut candidates = letters.difference(&leadings).cloned().collect::<Vec<_>>();
candidates.extend(" ".repeat(10 - candidates.len() - leadings.len()).chars());
let mut arrangement = AlphaNumMap::new();
let check = |map: &AlphaNumMap| {
let sum = terms.iter()
.map(|term| term.to_u64(map))
.sum::<u64>();
let result = result.to_u64(map);
sum == result
};
find_arrrangement(candidates, leadings, &mut arrangement, check);
arrangement.to_hashmap()
}
fn find_arrrangement<F: Copy + Fn(&AlphaNumMap) -> bool>(candidates: Vec<char>, leadings: HashSet<char>, arrangement: &mut AlphaNumMap, f: F) -> bool {
if candidates.is_empty() {
f(arrangement)
} else {
for (i,&e) in candidates.iter().enumerate() {
let mut subcandidates: Vec<char> = Vec::with_capacity(candidates.len());
subcandidates.extend(&candidates[..i]);
subcandidates.extend(&candidates[i+1..]);
subcandidates.extend(&leadings);
arrangement.push(e);
if find_arrrangement(subcandidates, HashSet::new(), arrangement, f) {
return true
}
arrangement.pop();
}
false
}
} Tags:
|
Exercise: anagramCodestruct Anagram{
lower: String,
sorted: Vec<char>
}
impl Anagram {
fn new(word: &str) -> Anagram {
let lower = word.to_lowercase();
let mut sorted: Vec<char> = lower.chars().collect::<Vec<_>>();
sorted.sort();
Anagram { lower: lower, sorted: sorted }
}
}
impl std::cmp::PartialEq for Anagram {
fn eq(&self, other: &Self) -> bool {
(self.lower != other.lower) && (self.sorted == other.sorted)
}
}
pub fn anagrams_for<'a>(word: &'a str, candidates: &[&'a str]) -> Vec<&'a str> {
let word_anagram = Anagram::new(word);
candidates.iter().filter_map(|&c|
if Anagram::new(c) == word_anagram { Some(c) } else { None }
).collect()
} Tags:
|
Exercise: robot-nameCodeextern crate rand;
use rand::distributions::Uniform;
use rand::{thread_rng, Rng};
pub struct Robot {
name: String,
}
impl Robot {
pub fn new() -> Self {
Robot {
name: Self::generate_name(),
}
}
fn generate_name() -> String {
let letter_part = thread_rng()
.sample_iter(&Uniform::new(b'A', b'Z'))
.take(2)
.map(|c| c as char)
.collect::<String>();
letter_part + "100"
}
pub fn name(&self) -> &str {
&self.name
}
pub fn reset_name(&mut self) {
self.name = Self::generate_name();
}
} Tags:
|
Exercise: parallel-letter-frequencyCodeuse std::collections::HashMap;
use rayon::prelude::*;
fn combine(left: HashMap<char,usize>,right: HashMap<char,usize>) -> HashMap<char,usize> {
let mut m = HashMap::new();
for (k,v) in left {
m.insert(k,v);
}
for (k,v) in right {
m.entry(k)
.and_modify(|v2| *v2 += v)
.or_insert(v);
}
return m
}
pub fn frequency(input: &[&str], worker_count: usize) -> HashMap<char, usize> {
match worker_count {
1 => seq_frequency(input),
_ => par_frequency(input)
}
}
fn process_str(s: &str) -> HashMap<char,usize> {
let mut m = HashMap::new();
for c in (*s).chars() {
normalize(c).map(|c| *m.entry(c).or_insert(0) += 1);
}
m
}
fn par_frequency(input: &[&str]) -> HashMap<char,usize> {
return input.par_iter()
.map(|s| process_str(s))
.reduce(|| HashMap::new(),|acc,x| combine(acc, x));
}
fn seq_frequency (input: &[&str]) -> HashMap<char, usize> {
let mut m = HashMap::new();
for s in input {
for c in (*s).chars() {
normalize(c).map(|c| *m.entry(c).or_insert(0) += 1);
}
}
return m
}
pub fn normalize(c: char) -> Option<char> {
if c.is_alphabetic() {
Some(c.to_ascii_lowercase())
} else {
None
}
} Tags:
|
Exercise: forthCodeuse std::collections::HashMap;
pub type Value = i32;
pub type ForthResult = Result<(), Error>;
pub type StackOperation = fn(&mut Vec<Value>) -> ForthResult;
pub enum Token {
Integer(Value),
Command(StackOperation),
}
impl Token {
pub fn apply(&self, to: &mut Vec<Value>) -> ForthResult {
match *self {
Token::Integer(n) => {
to.push(n);
Ok(())
},
Token::Command(f) => f(to),
}
}
}
impl Clone for Token {
fn clone(&self) -> Self {
match *self {
Token::Integer(ref n) => Token::Integer(*n),
Token::Command(ref f) => Token::Command(*f),
}
}
}
pub struct Forth(Vec<Value>, HashMap<String, Vec<Token>>);
#[derive(Debug, PartialEq)]
pub enum Error {
DivisionByZero,
StackUnderflow,
UnknownWord,
InvalidWord,
}
fn op_dup(stack: &mut Vec<Value>) -> ForthResult {
let n = stack.len();
if n==0 {
Err(Error::StackUnderflow)
} else {
let i = stack[n-1].clone();
stack.push(i);
Ok(())
}
}
fn op_drop(stack: &mut Vec<Value>) -> ForthResult {
match stack.pop() {
Some(_) => Ok(()),
None => Err(Error::StackUnderflow),
}
}
fn op_swap(stack: &mut Vec<Value>) -> ForthResult {
let n = stack.len();
if n<2 {
Err(Error::StackUnderflow)
} else {
let temp = stack.remove(n-2);
stack.push(temp);
Ok(())
}
}
fn op_over(stack: &mut Vec<Value>) -> ForthResult {
let n = stack.len();
if n<2 {
Err(Error::StackUnderflow)
} else {
let temp = stack[n-2].clone();
stack.push(temp);
Ok(())
}
}
fn op_plus(stack: &mut Vec<Value>) -> ForthResult {
if stack.len()<2 {
Err(Error::StackUnderflow)
} else {
let temp = stack.pop().unwrap() + stack.pop().unwrap();
stack.push(temp);
Ok(())
}
}
fn op_minus(stack: &mut Vec<Value>) -> ForthResult {
if stack.len()<2 {
Err(Error::StackUnderflow)
} else {
let temp = -stack.pop().unwrap() + stack.pop().unwrap();
stack.push(temp);
Ok(())
}
}
fn op_times(stack: &mut Vec<Value>) -> ForthResult {
if stack.len()<2 {
Err(Error::StackUnderflow)
} else {
let temp = stack.pop().unwrap() * stack.pop().unwrap();
stack.push(temp);
Ok(())
}
}
fn op_div(stack: &mut Vec<Value>) -> ForthResult {
if stack.len()<2 {
Err(Error::StackUnderflow)
} else {
match stack.pop().unwrap() {
0 => Err(Error::DivisionByZero),
ref n => {
let temp = stack.pop().unwrap() / *n;
stack.push(temp);
Ok(())
},
}
}
}
impl Forth {
pub fn new() -> Forth {
let mut ops: HashMap<String, Vec<Token>> = HashMap::new();
ops.insert("DUP".to_string(), vec![Token::Command(op_dup)]);
ops.insert("DROP".to_string(), vec![Token::Command(op_drop)]);
ops.insert("SWAP".to_string(), vec![Token::Command(op_swap)]);
ops.insert("OVER".to_string(), vec![Token::Command(op_over)]);
ops.insert("+".to_string(), vec![Token::Command(op_plus)]);
ops.insert("-".to_string(), vec![Token::Command(op_minus)]);
ops.insert("*".to_string(), vec![Token::Command(op_times)]);
ops.insert("/".to_string(), vec![Token::Command(op_div)]);
Forth(Vec::new(), ops)
}
pub fn format_stack(&self) -> String {
match self.0.len() {
0 => "".to_string(),
_ => {
let init = self.0[0].to_string();
self.0[1..].into_iter().fold(init, |b,v| b+" "+&v.to_string())
},
}
}
pub fn eval(&mut self, input: &str) -> ForthResult {
let terms: Vec<String> = input.split(|c: char| c.is_whitespace() || c.is_control()).map(|s| s.to_uppercase()).collect();
let mut i: usize = 0;
while i<terms.len() {
if terms[i].len()==0 {
i+=1;
} else if terms[i]==":".to_string() {
i+=1;
match self.eval_defn(&mut i, &terms) {
Ok(_) => (),
Err(res) => return Err(res),
};
} else {
match self.eval_term(&terms[i]) {
Ok(_) => i+=1,
Err(res) => return Err(res),
};
}
}
Ok(())
}
fn eval_defn(&mut self, i: &mut usize, terms: &Vec<String>) -> ForthResult {
if terms.len()<*i+3 || terms[*i].parse::<Value>().is_ok() {
Err(Error::InvalidWord)
} else {
let new_term = terms[*i].to_string();
let mut ops: Vec<Token> = Vec::new();
*i+=1;
while *i < terms.len() {
if terms[*i]==";" {
self.1.insert(new_term, ops);
*i+=1;
return Ok(());
}
match terms[*i].parse::<Value>() {
Ok(n) => ops.push(Token::Integer(n)),
Err(_) => match self.1.get(&terms[*i]) {
None => return Err(Error::UnknownWord),
Some(v) => ops.extend_from_slice(v),
},
};
*i+=1;
}
Err(Error::InvalidWord)
}
}
fn eval_term(&mut self, term: &String) -> ForthResult {
match term.parse::<Value>() {
Ok(n) => {
self.0.push(n);
Ok(())
},
Err(_) => match self.1.get(term) {
Some(v) => {
for t in v {
match t.apply(&mut self.0) {
Ok(_) => (),
Err(res) => return Err(res),
};
}
Ok(())
},
None => Err(Error::UnknownWord),
},
}
}
} Tags:
|
Exercise: forthCodeuse std::collections::HashMap;
pub type ValueType = i64;
pub type ForthResult = Result<(), Error>;
type ApplyResult = Result<Vec<ValueType>, Error>;
#[derive(Clone)]
enum Token {
Value(ValueType),
Call(String),
Def(Vec<Token>),
EndDef,
}
pub struct Forth {
stack: Vec<ValueType>,
custom_ops: HashMap<String, Vec<Token>>,
}
#[derive(Debug, PartialEq)]
pub enum Error {
DivisionByZero,
StackUnderflow,
UnknownWord,
InvalidWord,
}
impl Forth {
pub fn new() -> Self {
Forth{ stack: vec![], custom_ops: HashMap::new() }
}
pub fn format_stack(&self) -> String {
self.stack.iter().map(|e| e.to_string()).collect::<Vec<_>>().join(" ")
}
pub fn eval(&mut self, input: &str) -> ForthResult {
let input = input.to_lowercase();
let mut words = input.split(|c: char| c.is_whitespace() || c.is_control())
.filter(|&word| word.len() > 0);
let tokens = self.parse(&mut words, vec![]);
self.execute_all(tokens)
}
fn parse(&self, words: &mut Iterator<Item=&str>, mut tokens: Vec<Token>) -> Vec<Token> {
match words.next() {
Some(":") => {
tokens.push(Token::Def(self.parse(words, vec![])));
self.parse(words, tokens)
},
Some(";") => {
tokens.push(Token::EndDef);
tokens
},
Some(x) => {
tokens.push(self.atom(x));
self.parse(words, tokens)
},
None => tokens
}
}
fn atom(&self, word: &str) -> Token {
match word.parse() {
Ok(v) => Token::Value(v),
Err(_) => Token::Call(word.to_string())
}
}
fn execute_all(&mut self, tokens: Vec<Token>) -> ForthResult {
for token in tokens {
try!(self.execute(token));
}
Ok(())
}
fn execute(&mut self, token: Token) -> ForthResult {
match token {
Token::EndDef => Err(Error::InvalidWord),
Token::Value(v) => self.push(v),
Token::Call(op) => self.call(op),
Token::Def(def) => self.add_definition(&def[..])
}
}
fn call(&mut self, name: String) -> ForthResult {
if self.custom_ops.contains_key(&name) {
let tokens = self.custom_ops.get(&name).unwrap().clone();
self.execute_all(tokens)
} else {
self.primitive(name)
}
}
fn primitive(&mut self, input: String) -> ForthResult {
match &input[..] {
"+" => self.apply_2(|x, y| Ok(vec![x + y])),
"-" => self.apply_2(|x, y| Ok(vec![x - y])),
"*" => self.apply_2(|x, y| Ok(vec![x * y])),
"/" => self.apply_2(|x, y| if y == 0 { Err(Error::DivisionByZero) } else { Ok(vec![x / y]) }),
"dup" => self.apply_1(|x| Ok(vec![x, x])),
"drop" => self.apply_1(|_| Ok(vec![])),
"swap" => self.apply_2(|x, y| Ok(vec![y, x])),
"over" => self.apply_2(|x, y| Ok(vec![x, y, x])),
_ => Err(Error::UnknownWord)
}
}
fn add_definition(&mut self, def: &[Token]) -> ForthResult {
match (def.first(), def.last()) {
// A definition must start with the name to define, and end with an EndDef token
(Some(&Token::Call(ref name)), Some(&Token::EndDef)) => {
self.custom_ops.insert(name.to_string(), def[1 .. def.len() - 1].to_vec());
Ok(())
},
_ => Err(Error::InvalidWord)
}
}
// Some shorthands
fn push(&mut self, v: ValueType) -> ForthResult {
self.stack.push(v);
Ok(())
}
fn pop(&mut self) -> ValueType {
self.stack.pop().unwrap()
}
// Some helper functions to make evaluating operators easier
fn apply_1<F> (&mut self, f: F) -> ForthResult where F: Fn (ValueType) -> ApplyResult {
if self.stack.len() < 1 { return Err(Error::StackUnderflow) }
let x = self.pop();
self.process_apply_result(f(x))
}
fn apply_2<F> (&mut self, f: F) -> ForthResult where F: Fn (ValueType, ValueType) -> ApplyResult {
if self.stack.len() < 2 { return Err(Error::StackUnderflow) }
let (y, x) = (self.pop(), self.pop());
self.process_apply_result(f(x, y))
}
fn process_apply_result (&mut self, res: ApplyResult) -> ForthResult {
res.map(|mut results| self.stack.append(&mut results))
}
} Tags:
|
Exercise: nucleotide-codonsCodepub fn parse<'a>(pairs: Vec<(&'a str, &'a str)>) -> Codons<'a> {
Codons { pairs: pairs.iter().map(|x| Codon::new(x)).collect() }
}
pub struct Codons<'a> {
pairs: Vec<Codon<'a>>,
}
impl<'a> Codons<'a> {
pub fn name_for(&self, shorthand: &str) -> Result<&str, ()> {
if shorthand.is_empty() {
return Err(());
}
self.pairs
.iter()
.find(|ref x| x.matches(shorthand))
.map(|ref x| x.name)
.ok_or(())
}
}
struct Codon<'a> {
codon: &'a str,
name: &'a str,
}
impl<'a> Codon<'a> {
fn new(raw: &(&'a str, &'a str)) -> Codon<'a> {
Codon {
codon: raw.0,
name: raw.1,
}
}
fn matches(&self, shorthand: &str) -> bool {
shorthand.chars().zip(self.codon.chars()).all(|(a, b)| {
match a {
'H' => vec!['A', 'C', 'T'],
'M' => vec!['A', 'C'],
'N' => vec!['A', 'C', 'G', 'T'],
'R' => vec!['A', 'G'],
'Y' => vec!['C', 'T'],
_ => vec![a],
}
.contains(&b)
})
}
} Tags:
|
Exercise: doubly-linked-listCode// this module adds some functionality based on the required implementations
// here like: `LinkedList::pop_back` or `Clone for LinkedList<T>`
// You are free to use anything in it, but it's mainly for the test framework.
mod pre_implemented;
pub struct LinkedList<T> {
root : Option<Box<Link<T>>>,
}
pub struct Cursor<'a, T> {
current : *mut Link<T>,
list : &'a mut LinkedList<T>,
}
pub struct Iter<'a, T> {
current : Option<&'a Link<T>>,
}
impl<T> LinkedList<T> {
pub fn new() -> Self {
LinkedList {
root : None,
}
}
pub fn len(&self) -> usize {
if let Some(root) = &self.root {
Self::count_until_end(root, 1)
} else {
0
}
}
/// Return a cursor positioned on the front element
pub fn cursor_front(&mut self) -> Cursor<'_, T> {
if let Some(root) = &mut self.root {
Cursor {
current : &mut **root as *mut Link<T>,
list : self,
}
} else {
self.create_empty_cursor()
}
}
/// Return a cursor positioned on the back element
pub fn cursor_back(&mut self) -> Cursor<'_, T> {
if let Some(root) = &mut self.root {
let last = Self::navigate_until_end(root);
Cursor {
current : last as *mut Link<T>,
list : self,
}
} else {
self.create_empty_cursor()
}
}
/// Return an iterator that moves from front to back
pub fn iter(&self) -> Iter<'_, T> {
let mut current = None;
if let Some(root) = &self.root {
current = Some(&**root);
}
Iter {
current,
}
}
fn create_empty_cursor(&mut self) -> Cursor<'_, T> {
Cursor {
current : std::ptr::null_mut(),
list : self,
}
}
fn navigate_until_end(current : &mut Link<T>) -> &mut Link<T>
{
if current.next.is_none() {
return current;
}
let next = &mut *current.next.as_mut().unwrap();
Self::navigate_until_end(next)
}
fn count_until_end(current : &Link<T>, count : usize) -> usize
{
if current.next.is_none() {
return count;
}
let next = &*current.next.as_ref().unwrap();
Self::count_until_end(next, count+1)
}
}
// the cursor is expected to act as if it is at the position of an element
// and it also has to work with and be able to insert into an empty list.
impl<T> Cursor<'_, T> {
/// Take a mutable reference to the current element
pub fn peek_mut(&mut self) -> Option<&mut T> {
if let Some(link) = self.peek() {
return Some(&mut link.value)
}
None
}
/// Move one position forward (towards the back) and
/// return a reference to the new position
pub fn next(&mut self) -> Option<&mut T> {
if let Some(link) = self.peek() {
if let Some(next) = &mut link.next {
self.current = &mut **next;
}
}
self.peek_mut()
}
/// Move one position backward (towards the front) and
/// return a reference to the new position
pub fn prev(&mut self) -> Option<&mut T> {
if let Some(link) = self.peek() {
let prev; unsafe { prev = link.prev.as_mut(); }
if let Some(prev) = prev {
self.current = prev;
}
}
self.peek_mut()
}
/// Remove and return the element at the current position and move the cursor
/// to the neighboring element that's closest to the back. This can be
/// either the next or previous position.
pub fn take(&mut self) -> Option<T> {
if let Some(link) = self.peek() {
let prev; unsafe { prev = link.prev.as_mut(); }
let mut next = link.next.take();
let current;
if let Some(prev) = prev {
//this is not the root of the list
current = prev.next.take();
if let Some(next) = next.as_mut() {
next.prev = prev;
self.current = &mut **next;
} else {
self.current = prev;
}
prev.next = next;
} else {
//this is the root of the list
current = self.list.root.take();
if let Some(next) = next.as_mut() {
next.prev = std::ptr::null_mut();
self.current = &mut **next;
} else {
self.current = std::ptr::null_mut();
}
self.list.root = next;
}
return Some(current.unwrap().value);
}
None
}
pub fn insert_after(&mut self, element: T) {
if let Some(link) = self.peek() {
let mut next = link.next.take();
let mut new = Box::new(Link::<T> {
value : element,
next : None,
prev : link,
});
if let Some(next) = &mut next {
next.prev = &mut *new;
}
new.next = next;
link.next = Some(new);
} else {
self.insert_first(element);
}
}
pub fn insert_before(&mut self, element: T) {
if let Some(link) = self.peek() {
let mut new = Box::new(Link::<T> {
value : element,
next : None,
prev : link.prev,
});
let prev; unsafe { prev = link.prev.as_mut(); }
link.prev = &mut *new;
if let Some(prev) = prev {
new.next = prev.next.take();
prev.next = Some(new);
} else {
new.next = self.list.root.take();
self.list.root = Some(new);
}
} else {
self.insert_first(element);
}
}
fn peek(&mut self) -> Option<&mut Link<T>> {
let current; unsafe { current = self.current.as_mut(); }
if let Some(link) = current {
return Some(link)
}
None
}
fn insert_first(&mut self, element: T) {
let mut root = Box::new(Link::<T> {
value : element,
next : None,
prev : std::ptr::null_mut(),
});
self.current = &mut *root;
self.list.root = Some(root);
}
}
impl<'a, T> Iterator for Iter<'a, T> {
type Item = &'a T;
fn next(&mut self) -> Option<&'a T> {
let mut ret = None;
let mut new = None;
if let Some(link) = &self.current {
ret = Some(&link.value);
if let Some(next) = &link.next {
new = Some(&**next);
}
}
self.current = new;
ret
}
}
struct Link<T> {
value : T,
next : Option<Box<Link::<T>>>,//last has no next
prev : *mut Link::<T>,//first has no prev
} Tags:
|
Challenge accepted :) |
@iHiD I'm very sorry, but I vastly underestimated the time this is going to take. I changed a couple of tags, then scrolled through the list of possible tags and solutions to tag, and realized I have no chance of completing this anytime soon. |
This is an automated comment Hello 👋 Next week we're going to start using the tagging work people are doing on these. If you've already completed the work, thank you! If you've not, but intend to this week, that's great! If you're not going to get round to doing it, and you've not yet posted a comment letting us know, could you please do so, so that we can find other people to do it. Thanks! |
I don't intend to work on this anytime soon, feel free to find someone else / open this for community contributions. |
Thanks for the help! We've updated the tags. |
Hello lovely maintainers 👋
We've recently added "tags" to student's solutions. These express the constructs, paradigms and techniques that a solution uses. We are going to be using these tags for lots of things including filtering, pointing a student to alternative approaches, and much more.
In order to do this, we've built out a full AST-based tagger in C#, which has allowed us to do things like detect recursion or bit shifting. We've set things up so other tracks can do the same for their languages, but its a lot of work, and we've determined that actually it may be unnecessary. Instead we think that we can use machine learning to achieve tagging with good enough results. We've fine-tuned a model that can determine the correct tags for C# from the examples with a high success rate. It's also doing reasonably well in an untrained state for other languages. We think that with only a few examples per language, we can potentially get some quite good results, and that we can then refine things further as we go.
I released a new video on the Insiders page that talks through this in more detail.
We're going to be adding a fully-fledged UI in the coming weeks that allow maintainers and mentors to tag solutions and create training sets for the neural networks, but to start with, we're hoping you would be willing to manually tag 20 solutions for this track. In this post we'll add 20 comments, each with a student's solution, and the tags our model has generated. Your mission (should you choose to accept it) is to edit the tags on each issue, removing any incorrect ones, and add any that are missing. In order to build one model that performs well across languages, it's best if you stick as closely as possible to the C# tags as you can. Those are listed here. If you want to add extra tags, that's totally fine, but please don't arbitrarily reword existing tags, even if you don't like what Erik's chosen, as it'll just make it less likely that your language gets the correct tags assigned by the neural network.
To summarise - there are two paths forward for this issue:
If you tell us you're not able/wanting to help or there's no comment added, we'll automatically crowd-source this in a week or so.
Finally, if you have questions or want to discuss things, it would be best done on the forum, so the knowledge can be shared across all maintainers in all tracks.
Thanks for your help! 💙
Note: Meta discussion on the forum
The text was updated successfully, but these errors were encountered: