Skip to content
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

Add Grep exercise #94

Merged
merged 2 commits into from
Sep 15, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
8 changes: 8 additions & 0 deletions config.json
Original file line number Diff line number Diff line change
Expand Up @@ -415,6 +415,14 @@
"prerequisites": [],
"difficulty": 7
},
{
"slug": "grep",
"name": "Grep",
"uuid": "eefaa478-9e56-40b5-a378-4af53008c71d",
"practices": [],
"prerequisites": [],
"difficulty": 8
},
{
"slug": "rest-api",
"name": "REST API",
Expand Down
27 changes: 27 additions & 0 deletions exercises/practice/grep/.docs/instructions.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,27 @@
# Instructions

Search files for lines matching a search string and return all matching lines.

The Unix [`grep`][grep] command searches files for lines that match a regular expression.
Your task is to implement a simplified `grep` command, which supports searching for fixed strings.

The `grep` command takes three arguments:

1. The string to search for.
2. Zero or more flags for customizing the command's behavior.
3. One or more files to search in.

It then reads the contents of the specified files (in the order specified), finds the lines that contain the search string, and finally returns those lines in the order in which they were found.
When searching in multiple files, each matching line is prepended by the file name and a colon (':').

## Flags

The `grep` command supports the following flags:

- `-n` Prepend the line number and a colon (':') to each line in the output, placing the number after the filename (if present).
- `-l` Output only the names of the files that contain at least one matching line.
- `-i` Match using a case-insensitive comparison.
- `-v` Invert the program -- collect all lines that fail to match.
- `-x` Search only for lines where the search string matches the entire line.

[grep]: https://pubs.opengroup.org/onlinepubs/9699919799/utilities/grep.html
97 changes: 97 additions & 0 deletions exercises/practice/grep/.meta/Example.roc
Original file line number Diff line number Diff line change
@@ -0,0 +1,97 @@
module [grep]

import "iliad.txt" as iliad : Str
import "midsummer-night.txt" as midsummerNight : Str
import "paradise-lost.txt" as paradiseLost : Str

grep : Str, List Str, List Str -> Result Str _
grep = \pattern, flags, fileNames ->
config = parseFlags? flags
files = collectFiles? fileNames
displayFileNames = List.len files > 1
List.joinMap files \file ->
when findMatches config pattern file.text is
[] -> []
_ if config.displayFileNames -> [file.name]
matches ->
List.map matches \{ line, index } ->
lineNumber =
if config.displayLineNumbers then
"$(index + 1 |> Num.toStr):"
else
""
fileName =
if displayFileNames then
"$(file.name):"
else
""
"$(fileName)$(lineNumber)$(line)"
|> Str.joinWith "\n"
|> Ok

findMatches : Config, Str, Str -> List { line : Str, index : U64 }
findMatches = \config, pattern, text ->
Str.split text "\n"
|> List.mapWithIndex \line, index ->
{ line, index }
|> List.keepIf \{ line } ->
(lineToMatch, patternToMatch) =
if config.ignoreCase then
(toLower line, toLower pattern)
else
(line, pattern)

matches =
if config.matchFullLines then
lineToMatch == patternToMatch
else
Str.contains lineToMatch patternToMatch

# Using != is equivalent to xor which inverts `matches`
config.invertResults != matches

toLower : Str -> Str
toLower = \str ->
Str.toUtf8 str
|> List.map \byte ->
if 'A' <= byte && byte <= 'Z' then
byte - 'A' + 'a'
else
byte
|> Str.fromUtf8
|> Result.withDefault ""

Config : {
displayLineNumbers : Bool,
displayFileNames : Bool,
ignoreCase : Bool,
matchFullLines : Bool,
invertResults : Bool,
}

parseFlags : List Str -> Result Config _
parseFlags = \flags ->
defaultConfig = {
displayLineNumbers: Bool.false,
displayFileNames: Bool.false,
ignoreCase: Bool.false,
matchFullLines: Bool.false,
invertResults: Bool.false,
}
List.walkTry flags defaultConfig \config, flag ->
when flag is
"-l" -> Ok { config & displayFileNames: Bool.true }
"-n" -> Ok { config & displayLineNumbers: Bool.true }
"-i" -> Ok { config & ignoreCase: Bool.true }
"-x" -> Ok { config & matchFullLines: Bool.true }
"-v" -> Ok { config & invertResults: Bool.true }
_ -> Err (UnknownFlag flag)

collectFiles : List Str -> Result (List { name : Str, text : Str }) _
collectFiles = \names ->
List.mapTry names \name ->
when name is
"midsummer-night.txt" -> Ok { name: "midsummer-night.txt", text: midsummerNight }
"iliad.txt" -> Ok { name: "iliad.txt", text: iliad }
"paradise-lost.txt" -> Ok { name: "paradise-lost.txt", text: paradiseLost }
_ -> Err (FileNotFound name)
19 changes: 19 additions & 0 deletions exercises/practice/grep/.meta/config.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,19 @@
{
"authors": [
"isaacvando"
],
"files": {
"solution": [
"Grep.roc"
],
"test": [
"grep-test.roc"
],
"example": [
".meta/Example.roc"
]
},
"blurb": "Search a file for lines matching a regular expression pattern. Return the line number and contents of each matching line.",
"source": "Conversation with Nate Foster.",
"source_url": "https://www.cs.cornell.edu/Courses/cs3110/2014sp/hw/0/ps0.pdf"
}
15 changes: 15 additions & 0 deletions exercises/practice/grep/.meta/template.j2
Original file line number Diff line number Diff line change
@@ -0,0 +1,15 @@
{%- import "generator_macros.j2" as macros with context -%}
{{ macros.canonical_ref() }}
{{ macros.header() }}

import {{ exercise | to_pascal }} exposing [grep]

{% for case in cases -%}
{% for innerCase in case["cases"] -%}
# {{ case["description"] }} - {{ innerCase["description"] }}
expect
result = {{ innerCase["property"] | to_camel }} {{ innerCase["input"]["pattern"] | to_roc }} {{ innerCase["input"]["flags"] | to_roc }} {{ innerCase["input"]["files"] | to_roc }}
result == Ok {{ innerCase["expected"] | join('\n') | to_roc_multiline_string | indent(8) }}

{% endfor %}
{% endfor %}
85 changes: 85 additions & 0 deletions exercises/practice/grep/.meta/tests.toml
Original file line number Diff line number Diff line change
@@ -0,0 +1,85 @@
# This is an auto-generated file.
#
# Regenerating this file via `configlet sync` will:
# - Recreate every `description` key/value pair
# - Recreate every `reimplements` key/value pair, where they exist in problem-specifications
# - Remove any `include = true` key/value pair (an omitted `include` key implies inclusion)
# - Preserve any other key/value pair
#
# As user-added comments (using the # character) will be removed when this file
# is regenerated, comments can be added via a `comment` key.

[9049fdfd-53a7-4480-a390-375203837d09]
description = "Test grepping a single file -> One file, one match, no flags"

[76519cce-98e3-46cd-b287-aac31b1d77d6]
description = "Test grepping a single file -> One file, one match, print line numbers flag"

[af0b6d3c-e0e8-475e-a112-c0fc10a1eb30]
description = "Test grepping a single file -> One file, one match, case-insensitive flag"

[ff7af839-d1b8-4856-a53e-99283579b672]
description = "Test grepping a single file -> One file, one match, print file names flag"

[8625238a-720c-4a16-81f2-924ec8e222cb]
description = "Test grepping a single file -> One file, one match, match entire lines flag"

[2a6266b3-a60f-475c-a5f5-f5008a717d3e]
description = "Test grepping a single file -> One file, one match, multiple flags"

[842222da-32e8-4646-89df-0d38220f77a1]
description = "Test grepping a single file -> One file, several matches, no flags"

[4d84f45f-a1d8-4c2e-a00e-0b292233828c]
description = "Test grepping a single file -> One file, several matches, print line numbers flag"

[0a483b66-315b-45f5-bc85-3ce353a22539]
description = "Test grepping a single file -> One file, several matches, match entire lines flag"

[3d2ca86a-edd7-494c-8938-8eeed1c61cfa]
description = "Test grepping a single file -> One file, several matches, case-insensitive flag"

[1f52001f-f224-4521-9456-11120cad4432]
description = "Test grepping a single file -> One file, several matches, inverted flag"

[7a6ede7f-7dd5-4364-8bf8-0697c53a09fe]
description = "Test grepping a single file -> One file, no matches, various flags"

[3d3dfc23-8f2a-4e34-abd6-7b7d140291dc]
description = "Test grepping a single file -> One file, one match, file flag takes precedence over line flag"

[87b21b24-b788-4d6e-a68b-7afe9ca141fe]
description = "Test grepping a single file -> One file, several matches, inverted and match entire lines flags"

[ba496a23-6149-41c6-a027-28064ed533e5]
description = "Test grepping multiples files at once -> Multiple files, one match, no flags"

[4539bd36-6daa-4bc3-8e45-051f69f5aa95]
description = "Test grepping multiples files at once -> Multiple files, several matches, no flags"

[9fb4cc67-78e2-4761-8e6b-a4b57aba1938]
description = "Test grepping multiples files at once -> Multiple files, several matches, print line numbers flag"

[aeee1ef3-93c7-4cd5-af10-876f8c9ccc73]
description = "Test grepping multiples files at once -> Multiple files, one match, print file names flag"

[d69f3606-7d15-4ddf-89ae-01df198e6b6c]
description = "Test grepping multiples files at once -> Multiple files, several matches, case-insensitive flag"

[82ef739d-6701-4086-b911-007d1a3deb21]
description = "Test grepping multiples files at once -> Multiple files, several matches, inverted flag"

[77b2eb07-2921-4ea0-8971-7636b44f5d29]
description = "Test grepping multiples files at once -> Multiple files, one match, match entire lines flag"

[e53a2842-55bb-4078-9bb5-04ac38929989]
description = "Test grepping multiples files at once -> Multiple files, one match, multiple flags"

[9c4f7f9a-a555-4e32-bb06-4b8f8869b2cb]
description = "Test grepping multiples files at once -> Multiple files, no matches, various flags"

[ba5a540d-bffd-481b-bd0c-d9a30f225e01]
description = "Test grepping multiples files at once -> Multiple files, several matches, file flag takes precedence over line number flag"

[ff406330-2f0b-4b17-9ee4-4b71c31dd6d2]
description = "Test grepping multiples files at once -> Multiple files, several matches, inverted and match entire lines flags"
9 changes: 9 additions & 0 deletions exercises/practice/grep/Grep.roc
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
module [grep]

import "iliad.txt" as iliad : Str
import "midsummer-night.txt" as midsummerNight : Str
import "paradise-lost.txt" as paradiseLost : Str

grep : Str, List Str, List Str -> Result Str _
grep = \pattern, flags, files ->
crash "Please implement 'grep'"
Loading