Попытка разбора / etc / passwd Использование Java 8 с потоками - PullRequest
0 голосов
/ 23 марта 2019

Я пытаюсь проанализировать файл / etc / passwd в операционной системе MacOS Mojave 10.14.3, используя потоковую функцию Java 1.8.

Первые пять строк в моем файле / etc / passwd выглядят следующим образом:

nobody:*:-2:-2:Unprivileged User:/var/empty:/usr/bin/false
root:*:0:0:System Administrator:/var/root:/bin/sh
daemon:*:1:1:System Services:/var/root:/usr/bin/false
_uucp:*:4:4:Unix to Unix Copy 
Protocol:/var/spool/uucp:/usr/sbin/uucico
_taskgated:*:13:13:Task Gate Daemon:/var/empty:/usr/bin/false

Вот моя реализация:

import java.nio.file.Files;
import java.nio.file.Paths;
import java.util.ArrayList;
import java.util.stream.Collectors;
import java.util.stream.Stream;

public class User {

    String name;
    Integer uid;
    Integer gid;
    String comment;
    String home;
    String shell;

    public User(String filePath) {
        String[] items = filePath.split(":");

        if (items.length != 7) {
            throw new IllegalArgumentException("Need 7 items from file and there's only: " + items.length);
        }

        this.name = items[0];
        this.uid = Integer.parseInt(items[2]);
        this.gid = Integer.parseInt(items[3]);
        this.comment = items[4];
        this.home = items[5];
        this.shell = items[6];
    }

    // Getters and Setters omitted for brevity.


    public static void main(String[] args) {
        // Read the file
        try (Stream<String> stream = Files.lines(Paths.get("/etc/passwd")))
        {
            ArrayList<User> users = stream.map(User::new).filter(u -> !u.shell.equals("/bin/false"))
                                                         .filter(u -> !u.shell.equals("/usr/sbin/nologin"))
                                                         .collect(Collectors.toCollection(ArrayList::new));
            for (User user : users) {
                System.out.println("User: " + user.name);
            }
        } 

        catch(Throwable t) {
            t.printStackTrace();
        }   
    }
}

IllegalArgumentException:

java.lang.IllegalArgumentException: Need 7 items from file and there's only: 1
    at com.sample.model.User.<init>(User.java:23)
    at java.util.stream.ReferencePipeline$3$1.accept(ReferencePipeline.java:193)
    at java.util.Iterator.forEachRemaining(Iterator.java:116)
    at java.util.Spliterators$IteratorSpliterator.forEachRemaining(Spliterators.java:1801)
    at java.util.stream.AbstractPipeline.copyInto(AbstractPipeline.java:481)
    at java.util.stream.AbstractPipeline.wrapAndCopyInto(AbstractPipeline.java:471)
    at java.util.stream.ReduceOps$ReduceOp.evaluateSequential(ReduceOps.java:708)
    at java.util.stream.AbstractPipeline.evaluate(AbstractPipeline.java:234)
    at java.util.stream.ReferencePipeline.collect(ReferencePipeline.java:499)
    at com.sample.model.User.main(User.java:88)

Вопрос (ы):

  1. Что я, возможно, делаю не так?

  2. Есть ли другой способ сделать это вместо использования функции потоков в Java 1.8? Похоже, что потоки сложнее отлаживать ...

1 Ответ

1 голос
/ 23 марта 2019

Я думаю, вам нужно учитывать строки, которые являются комментариями / начинаются с '#'.

Итак:

ArrayList<User> users = stream.filter(s -> s.charAt(0) != '#')
                              .map(User::new)
                              .filter(u -> !u.shell.equals("/bin/false"))
                              .filter(u -> !u.shell.equals("/usr/sbin/nologin"))
                              .collect(Collectors.toCollection(ArrayList::new));

С вашей точки зрения, re: отладка - да, инструменты не очень хороши (пока). Возможно, вы могли бы это уловить, проверив filePath, переданный конструктору User, и заметив, что это был комментарий.

...