-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathExercise-old.hs
631 lines (553 loc) · 18.9 KB
/
Exercise-old.hs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
{-# LANGUAGE CPP #-}
{-# LANGUAGE Safe #-}
module Exercise where
-- Exercise set 2.
--
-- 30% of the exercises are intended to be rather challenging, and
-- will allow you to get a mark above 69%, in conjunction with the
-- other available exercises, so as to get a 1st class mark. To get
-- II.2, you will need to do enough hard exercises, in addition to the
-- medium and easy ones.
--
-- >= 70% 1st
-- >= 60% II.1
-- >= 50% II.2
-- >= 40% III
-- <= 39% fail
--
--
-- Do as many unassessed exercises as you can, as they should make the
-- assessed exercises easier.
--
-- You are allowed to use the functions available in the standard
-- prelude (loaded by default by ghc and ghci). You should not need to
-- use other Haskell libraries from the Haskell platform available in
-- the lab, but you are allowed to use them if you wish. However, in
-- your final submission, you should not use any IO facilities, as
-- this won't compile with the marking script.
-- This exercise set revolves around a directory tree on a computer,
-- and some Unix-like functions to manipulate them.
--
-- This exercise doesn't involve reading or writing actual files from
-- disk. Instead, we represent them internally in Haskell using a
-- "data" definition, explained below.
--
-- We have these data types:
--
-- - Entry: Can be either a file or a directory (with sub-Entries)
-- - EntryName: exactly the same as a string;
-- represents a directory or file name
-- - Path: exactly the same as a list of strings
-- - FileProp: file properties
data Entry = File EntryName FileProp
| Dir EntryName [Entry]
deriving (Show, Eq, Read)
-- The name of a file
type EntryName = String
-- A sequence of file name components.
--
-- A path is a list of strings used to navigate down to a subdirectory
-- of a given directory. We start from an Entry, and we end up with a
-- sub-Entry, provided the path is valid. See the example in the "cd"
-- exercise below.
type Path = [String]
-- FileProp describes some attributes of a file.
--
-- The components mean size, content, and creation time,
-- in that order.
data FileProp = FP Int String Int
deriving (Show, Eq, Read)
-- Exercise, easy. Create a FileProp that describes a file with size 3,
-- content "abc", and time 4.
exampleFP :: FileProp
exampleFP = FP 3 "abc" 4
{-
Entries describe directories and files in the file system. For instance, the
following entry describes an empty directory:
Dir "somedirname" []
The following entry describes a file in the filesystem:
File "somefilename" (FP 4 "xyz" 3)
The following entry describes a more complicated directory.
Dir "uni" [File "marks.txt" (FP 1036 "..." 2014),
Directory "examSheets" [],
File "address.txt" (FP 65 "..." 2010)
]
-}
-- Exercise, easy. Create three entries that correspond to the following trees:
--
-- 1. todo.txt, size 723, time 2015, content "do fp exercises"
--
-- 2. empty-directory
-- |
--
-- 3. hard drive
-- |
-- |-- WINDOWS
-- | |
-- | |-- cmd.exe, size 1024, time 1995, content ""
-- | |
-- | |-- explorer.exe, size 2048, time 1995, content ""
-- |
-- |-- Documents
-- | |
-- | |-- User1
-- | | |
-- | | |-- recipe.doc, size 723, time 2000
-- | |
-- | |-- User2
-- | | |
--
-- You must pay attention to the order of entries in a directory.
--
-- There is a dash in the directory name of exampleEntry2.
exampleEntry1 :: Entry
exampleEntry1 = File "todo.txt" (FP 1024 "do fp exercises" 2015)
exampleEntry2 :: Entry
exampleEntry2 = Dir "empty-directory" []
exampleEntry3 :: Entry
exampleEntry3 = Dir "hard drive" [Dir "WINDOWS" [File "cmd.exe" (FP 1024 "" 1995), File "explorer.exe" (FP 2048 "" 1995)], Dir "Documents" [Dir "User1" [File "recipe.doc" (FP 723 "" 2000)], Dir "User2" []]]
-- Exercise, unassessed. You're given a directory as a value of type
-- Entry. In this directory there is a subdirectory with name n. Find
-- (and return) this subdirectory.
cd1 :: Entry -> String -> Maybe Entry
cd1 (File _ _ ) _ = Nothing
cd1 (Dir a ys) n = case ys of
[] -> Nothing
(x:xs) -> case x of
(File _ _) -> cd1 (Dir a xs) n
(Dir x' ys') |x' == n -> Just(Dir x' ys')
|otherwise -> cd1 (Dir a xs) n
-- Exercise, easy. As before, but you need to navigate not one but
-- possibly many steps down; consecutive directory names are given in
-- the list of strings.
--
-- Example: Given the entry in the following drawing
--
-- root
-- |
-- |-- dir1
-- | |
-- | |-- dir1a
-- | | |
-- | | |-- dir1a1
-- | | |
-- | | |-- dir1a2
-- | |
-- | |-- dir1b
-- |
-- |-- dir2
-- | |
-- | |-- dir1a
-- | |
-- | |-- dir1b
-- |
-- |-- file3
--
-- and the path ["dir1", "dir1a"], you need to return the Entry that
-- contains dir1a1 and dir1a2.
--
-- If there is no such entry, return Nothing. If there is such an entry,
-- return Just that entry.
--
-- You can assume that there will be at most one entry with the given path.
cd :: Entry -> Path -> Maybe Entry
cd (File _ _) _ = Nothing
cd all@(Dir a ys) [] = Just(all)
cd (Dir a ys) path@(x:xs) = case ys of
[] -> Nothing
(y:ys) -> case y of
(File _ _) -> cd (Dir a ys) path
(Dir i zs) |i == x -> cd (Dir i zs) xs
|otherwise -> cd (Dir a ys) path
-- Exercise, medium. Split a string representing a path into its
-- components. The components are separated by (forward) slashes.
-- Hint: the prelude functions for lists will be helpful here, but you
-- are not required to use them.
--
-- Examples:
--
-- explode "abc/de/fghi" = ["abc", "de", "fghi"]
-- explode "abc//fghi" = ["abc", "", "fghi"]
--
-- It is a matter of convention whether we prefer
-- explode "" = []
-- or
-- explode "" = [""]
-- Choose your own convention. Both will be considered to be correct.
explode :: String -> Path
explode "" = []
explode (x:xs) | x == '/' = "":explode xs
| otherwise = case (explode xs) of
[] -> [[x]]
(y:ys) -> (x:y):ys
-- Exercise, easy. The "inverse" of explode: combine components with
-- slashes to a single String.
--
-- For every string s, you must have
--
-- implode (explode s) = s
--
-- You may want to use the functions "concat", "intersperse", and/or
-- "intercalate"; the latter two are from package Data.List.
implode :: Path -> String
implode [] = ""
implode [x] = x
implode (x:xs) = x++('/':implode xs)
-- Exercise, easy. Given an Entry representing a directory, print out
-- a directory listing in a format similar to "ls -l" on Unix.
--
-- The required format is as in the following example:
--
-- size: 420 time: 5 filename1
-- size: 5040 time: 200 other.txt
-- size: 30 time: 36 filename2
--
-- You need to separate every line with a newline ('\n') character,
-- and also put a newline at the end.
--
-- Keep the files in their order in the given Entry.
--
-- You do not need to convert units, just print the numbers.
lsL :: Entry -> String
lsL (Dir a ys) = case ys of
[] -> ""
(x:xs) -> case x of
(File name (FP size _ time)) -> "size: " ++ (show size) ++ " time: " ++ (show time) ++ " " ++ (show name) ++ "\n" ++lsL(Dir a xs)
(Dir b _) -> b ++ "\n" ++ lsL (Dir a xs)
-- Exercise, medium. List all the files in a directory tree. Sample
-- output:
--
-- root
-- |
-- |-- dir1
-- | |
-- | |-- dir1a
-- | | |
-- | | |-- dir1a1
-- | | |
-- | | |-- somefile
-- | | |
-- | | |-- dir1a2
-- | |
-- | |-- dir1b
-- |
-- |-- file2
-- |
-- |-- dir3
-- | |
-- | |-- dir3a
-- | |
-- | |-- dir3b
--
--
-- You can assume that the entry represents a directory.
--
-- Use the newline convention as given above.
dupl :: (Num a,Eq a) => a -> String -> String
dupl 0 _ = ""
dupl n xs = xs ++ dupl (n-1) xs
lsTree :: Entry -> String
--lsTree (Dir a []) = a ++ "\n|\n"--MAY NEED TO CHANGE!!!!
lsTree (Dir a ys) = let lines n (Dir a ys) = a ++ let tree2 ls = case ls of
[] -> ""
(x:xs) -> (dupl 2 ("\n|" ++ (dupl n " |"))) ++"-- " ++ case x of
(File a _) -> a ++ (tree2 xs)
--(Dir b []) -> b ++"\n|" ++ (dupl (n+1) " |")++(tree2 xs) MAY NEED TO CHANGE
(Dir b zs) -> (lines (n+1) (Dir b zs)) ++ (tree2 xs)
in tree2 ys
in lines 0 (Dir a ys) ++ "\n"
-- Exercise, challenge. Make a list of all the files and directories
-- recursively in a tree (similar to "find ." in linux). If the
-- argument fullPath is False, every entry in the returned list will
-- have only the bare directory or file name. If fullPath is True,
-- every entry is the path towards that entry,
-- e.g. "root/subdir1/subdir1a/file".
--
-- The root must be the first list item. The output will be in the
-- same order as for lsTree.
--
-- For example, if d is this directory from an earlier exercise:
--
-- hard drive
-- |
-- |-- WINDOWS
-- | |
-- | |-- cmd.exe, size 1024, time 1995, content ""
-- | |
-- | |-- explorer.exe, size 2048, time 1995, content ""
-- |
-- |-- Documents
-- | |
-- | |-- User1
-- | | |
-- | | |-- recipe.doc, size 723, time 2000
-- | |
-- | |-- User2
-- | | |
--
-- then we have
-- listAll False d =
-- ["hard drive", "WINDOWS", "cmd.exe", "explorer.exe"
-- ,"Documents", "User1", "recipe.doc", "User2"]
-- and
-- listAll True d =
-- ["hard drive"
-- ,"hard drive/WINDOWS"
-- ,"hard drive/WINDOWS/cmd.exe"
-- ,"hard drive/WINDOWS/explorer.exe"
-- ,"hard drive/Documents"
-- ,"hard drive/Documents/User1"
-- ,"hard drive/Documents/User1/recipe.doc",
-- ,"hard drive/Documents/User2"]
listAll :: Bool -> Entry -> [String]
listAll _ (File a _) = [a]
listAll False (Dir a ys) = a : let listAllList ls = case ls of
[] -> []
(x:xs) -> case x of
(File a _) -> (a : listAllList xs)
(Dir b zs) -> (listAll False (Dir b zs)) ++ (listAllList xs)
in listAllList ys
listAll True (Dir a ys) = let listAll' path (Dir a ys) = implode(path++[a]):let wList ls = case ls of
[] -> []
(x:xs) -> case x of
(File b _) -> (implode(path++[a])++"/"++b):(wList xs)
(Dir b zs) -> (listAll' (path++[a])(Dir b zs))++(wList xs)
in wList ys
in listAll' [] (Dir a ys)
-- Exercise, hard.
--
-- Given a tree, insert a given subtree in a certain position.
--
-- It does not matter how the inserted subtree is ordered with respect
-- to the other items in the directory where it is inserted. That is,
--
-- cp (Dir "root" [Dir "subdir1" [Dir "subdir1a" []]]) (["subdir1"], Dir "subdir1b" [])
--
-- may return either
--
-- Dir "root" [Dir "subdir1" [Dir "subdir1a" [], Dir "subdir1b" []]]
--
-- or
--
-- Dir "root" [Dir "subdir1" [Dir "subdir1b" [], Dir "subdir1a" []]] .
--
-- (This function is similar-ish to the Unix 'cp' utility.)
cp :: Entry -> (Path, Entry) -> Entry
cp (File _ _) _ = undefined
cp (Dir a ys) ([],e) = Dir a (e:ys)
cp (Dir a ys) ((x:xs),e) = let cp' ls = case ls of
[] -> []
(z:zs) -> case z of
(File b f) -> z:(cp' ls)
(Dir b xs') | b == x -> (cp z (xs,e)):zs
| otherwise -> z:(cp' zs)
in Dir a (cp' ys)
-- Exercise, medium. Given a tree and a path, remove the file or
-- directory at that path.
--
-- You can assume that there is a file or directory at that path. If
-- there are multiple files or directories with that path, you need to
-- remove all of them.
--
-- (In that the case, the tree would not be "valid" according to isValid.)
-- NEED TO CHECK WITH SOMEONE!!!
rm :: Entry -> Path -> Entry
rm (File _ _) _ = undefined
rm (Dir a ys) [] = Dir a ys --empty path so return directory
rm (Dir a ys) (x:xs) = let rm' ls = case ls of
[] -> []
(z:zs) -> case z of
(File b f) | xs == [] && x == b -> rm' zs
| otherwise -> z:(rm' zs)
(Dir b xs') | b==x && xs == [] -> rm' zs
| b==x -> (rm z xs):zs
| otherwise -> z:(rm' zs)
in Dir a (rm' ys)
-- Exercise, harder. Return a tree with all the same entries, but so
-- that the entries of each (sub)directory are in sorted order.
--
-- You may use the function `sort` from the Prelude.
--
-- If there are multiple entries with the same name in a directory,
-- you may choose any order.
--CHECK SORTED IN CORRECT WAY!!!
getName::Entry -> EntryName
getName (File a _) = a
getName (Dir a ys) = a
sortWith :: (Ord b) => (a->b) -> [a] -> [a]
sortWith _ [] = []
sortWith f (x:xs) = smaller ++ [x] ++ bigger
where smaller = sortWith f [a | a <- xs, f a <= f x]
bigger = sortWith f [a | a <- xs, f a > f x]
sortTree :: Entry -> Entry
sortTree (Dir a ys) = let sortTree' ls = case ls of
[] -> []
(z:zs) -> case z of
(File a f) -> (File a f):sortTree' zs
(Dir b xs') -> (sortTree z):sortTree' zs
in Dir a (sortTree' (sortWith getName ys))
-- Exercise, unassessed. Change all letters to upper case in a string.
--
-- For instance,
--
-- upcaseStr "!someString123" = "!SOMESTRING123"
--
-- Hint: look at the definition of the String type in the Prelude, and think
-- about functions related to that type.
--
-- You may use the function upcaseChar, defined below.
upcaseStr :: String -> String
upcaseStr [] = []
upcaseStr (x:xs)
|x `elem` ['a'..'z'] = (upcaseChar x):(upcaseStr xs)
|otherwise = x:(upcaseStr xs)
upcaseChar :: Char -> Char
upcaseChar c =
if ('a' <= c && c <= 'z')
then toEnum (fromEnum c + fromEnum 'A' - fromEnum 'a')
else c
-- Exercise, harder. Change all the file names (as above) and their
-- properties, similar to the above exercise.
--
-- From the type of modifyEntries, you can see what the input of
-- fileMap must be.
flip' :: (a -> b -> c) -> (b -> a -> c)
flip' f a b = f b a
modifyEntries :: Entry -> ((EntryName, FileProp) -> (EntryName, FileProp)) -> Entry
modifyEntries (File a f) fileMap = let (b,c) = fileMap (a,f) in (File b c)
modifyEntries (Dir a ys) fileMap = case ys of
[] -> Dir a ys
(z:zs) -> Dir a (map (flip' modifyEntries fileMap) ys)
-- Exercise, unassessed. Create a "Fibonacci tree".
--
-- The Fibonacci tree for n=3 looks like this:
--
-- dir3
-- |
-- |-- dir2
-- | |
-- | |-- dir1
-- | | |
-- | | |-- file, size 0, time 0, content 0
-- | |
-- | |
-- | |-- file, size 0, time 0, content 0
-- |
-- |-- dir1
-- | |
-- | |-- file, size 0, time 0, content 0
--
--
--
-- The Fibonacci tree (fibCreate 0) is a file with name "file", size and time
-- 0, and content "". For n >= 1, fibCreate n is a directory named "dir{n}",
-- containing precisely fibCreate (n-1) and fibCreate (n-2). Exception:
-- fibCreate 1 contains only fibCreate 0, and not fibCreate (-1).
--
-- (We just made up the concept of Fibonacci trees, it is not a real thing.)
fibCreate :: Int -> Entry
fibCreate 0 = File "file" (FP 0 "" 0)
fibCreate 1 = Dir "dir1" [fibCreate 0]
fibCreate n = Dir ("dir" ++ show n) ([fibCreate (n-1)]++[fibCreate (n-2)])
-- Exercise, unassessed. Make the following infinite tree:
--
-- all
-- |
-- |-- file, size 0, time 0, content 0
-- |
-- |-- dir1
-- | |
-- | |-- file, size 0, time 0, content 0
-- |
-- |-- dir2
-- | |
-- | |-- dir1
-- | | |
-- | | |-- file, size 0, time 0, content 0
-- |
-- |-- dir3
-- | |
-- | |-- dir2
-- | | |
-- | | |-- dir1
-- | | | |
-- | | | |-- file, size 0, time 0, content 0
-- | |
-- | |-- dir1
-- | | |
-- | | |-- file, size 0, time 0, content 0
-- |
-- |-- dir4
-- | |
-- | (and so on)
-- |
-- | ...
--
--
-- It is to be expected that computations such as (size fibEntry) will
-- not return a result and loop for ever (or until we run out of
-- memory). But you can still e.g. "cd" into such a tree.
fibEntry :: Entry
fibEntry = Dir "all" [(fibCreate x)|x <- [0..]]
-- Exercise, unassessed. Remove from a tree all files that are larger
-- than a certain size. You should not remove any directories.
--
-- Files that are exactly that size should be kept in.
findSmallerThan :: Entry -> Int -> Entry
findSmallerThan (Dir a ys) maxSize = let check ls = case ls of
[] -> []
(z:zs) -> case z of
File a (FP b c d)
| b > maxSize -> check zs
| otherwise -> (File a (FP b c d)):check zs
Dir b xs' -> (findSmallerThan (Dir b xs') maxSize):check zs
in Dir a (check ys)
-- Exercise, challenge. Remove from a tree all files that do not
-- satisfy a given predicate. You should not remove any directories.
find :: Entry -> ((EntryName, FileProp) -> Bool) -> Entry
find (Dir a ys) predicate = let check ls = case ls of
[] -> []
(z:zs) -> case z of
File a f
| predicate (a,f) -> (File a f):check zs
| otherwise -> check zs
Dir b xs' -> (find (Dir b xs') predicate):check zs
in Dir a (check ys)
-- Exercise, unassessed. Given a maximum file size, a file name and its file
-- properties, return whether the file is at most that size.
--
-- (This function gets a lot of information that it doesn't need; the
-- extra arguments are so that you can easily use `findSmallerThanPred
-- maxSize` as the predicate argument to `find`, in the next
-- exercise.)
findSmallerThanPred :: Int -> ((EntryName, FileProp) -> Bool)
findSmallerThanPred maxSize (filename, (FP size _ _)) = size <= maxSize
-- Exercise, unassessed. Same as findSmallerThan, but implement it again
-- using `find` and `findSmallerThanPred`.
findSmallerThan2 :: Entry -> Int -> Entry
findSmallerThan2 root maxSize = find root (findSmallerThanPred maxSize)
-- Exercise, challenge, assessed.
--
-- List all directory and file names in the current directory in a
-- table. The table can be at most termWidth cells wide. You need to
-- use as few rows as possible, while separating columns with 2
-- spaces.
--
-- (This is similar to the Unix utility 'ls'.)
--
-- For instance, for terminal width 80, you might have the following
-- output:
--
-- a d g j zbcdefghijklmnc zbcdefghijklmnf zbcdefghijklmni
-- b e h zbcdefghijklmna zbcdefghijklmnd zbcdefghijklmng
-- c f i zbcdefghijklmnb zbcdefghijklmne zbcdefghijklmnh
--
--
-- The ordering is alphabetical by column, and the columns should be
-- indented as above. You can assume that the longest directory/file
-- name is at most as long as the terminal is wide.
--
-- The first argument is the terminal width:
ls :: Int -> Entry -> String
ls termWidth root = undefined
-- End of exercise set 2.