提交修改
This commit is contained in:
parent
f24deffe00
commit
894fc29901
23
src/main/java/com/lz/common/cli/CliToken.java
Normal file
23
src/main/java/com/lz/common/cli/CliToken.java
Normal file
@ -0,0 +1,23 @@
|
|||||||
|
package com.lz.common.cli;
|
||||||
|
|
||||||
|
public interface CliToken {
|
||||||
|
/**
|
||||||
|
* @return the token value
|
||||||
|
*/
|
||||||
|
String value();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return the raw token value, that may contain unescaped chars, for instance {@literal "ab\"cd"}
|
||||||
|
*/
|
||||||
|
String raw();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return true when it's a text token
|
||||||
|
*/
|
||||||
|
boolean isText();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return true when it's a blank token
|
||||||
|
*/
|
||||||
|
boolean isBlank();
|
||||||
|
}
|
||||||
43
src/main/java/com/lz/common/cli/CliTokens.java
Normal file
43
src/main/java/com/lz/common/cli/CliTokens.java
Normal file
@ -0,0 +1,43 @@
|
|||||||
|
package com.lz.common.cli;
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
import com.lz.common.cli.impl.CliTokenImpl;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author beiwei30 on 09/11/2016.
|
||||||
|
*/
|
||||||
|
public class CliTokens {
|
||||||
|
/**
|
||||||
|
* Create a text token.
|
||||||
|
*
|
||||||
|
* @param text the text
|
||||||
|
* @return the token
|
||||||
|
*/
|
||||||
|
public static CliToken createText(String text) {
|
||||||
|
return new CliTokenImpl(true, text, text);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a new blank token.
|
||||||
|
*
|
||||||
|
* @param blank the blank value
|
||||||
|
* @return the token
|
||||||
|
*/
|
||||||
|
public static CliToken createBlank(String blank) {
|
||||||
|
return new CliTokenImpl(false, blank, blank);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tokenize the string argument and return a list of tokens.
|
||||||
|
*
|
||||||
|
* @param s the tokenized string
|
||||||
|
* @return the tokens
|
||||||
|
*/
|
||||||
|
public static List<CliToken> tokenize(String s) {
|
||||||
|
return CliTokenImpl.tokenize(s);
|
||||||
|
}
|
||||||
|
}
|
||||||
8
src/main/java/com/lz/common/cli/IntConsumer.java
Normal file
8
src/main/java/com/lz/common/cli/IntConsumer.java
Normal file
@ -0,0 +1,8 @@
|
|||||||
|
package com.lz.common.cli;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author bw on 25/10/2016.
|
||||||
|
*/
|
||||||
|
public interface IntConsumer {
|
||||||
|
void accept(int value);
|
||||||
|
}
|
||||||
160
src/main/java/com/lz/common/cli/LineStatus.java
Normal file
160
src/main/java/com/lz/common/cli/LineStatus.java
Normal file
@ -0,0 +1,160 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2015 Julien Viet
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package com.lz.common.cli;
|
||||||
|
|
||||||
|
|
||||||
|
import java.util.LinkedList;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author <a href="mailto:julien@julienviet.com">Julien Viet</a>
|
||||||
|
*/
|
||||||
|
public class LineStatus implements IntConsumer {
|
||||||
|
|
||||||
|
// Keeping this internal to this package
|
||||||
|
static class Ext extends LineStatus implements IntConsumer {
|
||||||
|
final LinkedList<Integer> buffer;
|
||||||
|
Ext() {
|
||||||
|
this.buffer = new LinkedList<Integer>();
|
||||||
|
}
|
||||||
|
public void accept(int codePoint) {
|
||||||
|
super.accept(codePoint);
|
||||||
|
switch (transition) {
|
||||||
|
case TO_WEAK:
|
||||||
|
buffer.add((int) '"');
|
||||||
|
break;
|
||||||
|
case TO_STRONG:
|
||||||
|
buffer.add((int) '\'');
|
||||||
|
break;
|
||||||
|
case FROM_WEAK:
|
||||||
|
buffer.add((int) '"');
|
||||||
|
break;
|
||||||
|
case FROM_STRONG:
|
||||||
|
buffer.add((int) '\'');
|
||||||
|
break;
|
||||||
|
case TO_ESC:
|
||||||
|
buffer.add((int)'\\');
|
||||||
|
break;
|
||||||
|
case FROM_ESC:
|
||||||
|
if (codePoint != '\r') {
|
||||||
|
buffer.add(codePoint);
|
||||||
|
} else {
|
||||||
|
buffer.removeLast();
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case CODE_POINT:
|
||||||
|
buffer.add(codePoint);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
protected int quote = 0;
|
||||||
|
protected Transition transition;
|
||||||
|
|
||||||
|
public boolean isEscaped() {
|
||||||
|
return transition == Transition.FROM_ESC;
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean isEscaping() {
|
||||||
|
return transition == Transition.TO_ESC;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return true if it's currently quoted
|
||||||
|
*/
|
||||||
|
public boolean isQuoted() {
|
||||||
|
return quote != 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean isWeaklyQuoted() {
|
||||||
|
return quote == '"';
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean isStronglyQuoted() {
|
||||||
|
return quote == '\'';
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return the current quote: {@code 0}, {@code '} or {@code "} value
|
||||||
|
*/
|
||||||
|
public int getQuote() {
|
||||||
|
return quote;
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean isCodePoint() {
|
||||||
|
return transition == Transition.CODE_POINT || transition == Transition.FROM_ESC;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void accept(int cp) {
|
||||||
|
Transition next;
|
||||||
|
if (transition == Transition.TO_ESC) {
|
||||||
|
next = Transition.FROM_ESC;
|
||||||
|
} else {
|
||||||
|
switch (quote) {
|
||||||
|
case 0:
|
||||||
|
switch (cp) {
|
||||||
|
case '\'':
|
||||||
|
quote = '\'';
|
||||||
|
next = Transition.TO_STRONG;
|
||||||
|
break;
|
||||||
|
case '"':
|
||||||
|
quote = '"';
|
||||||
|
next = Transition.TO_WEAK;
|
||||||
|
break;
|
||||||
|
case '\\':
|
||||||
|
next = Transition.TO_ESC;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
next = Transition.CODE_POINT;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case '\'':
|
||||||
|
if (cp == '\'') {
|
||||||
|
quote = 0;
|
||||||
|
next = Transition.FROM_STRONG;
|
||||||
|
} else {
|
||||||
|
next = Transition.CODE_POINT;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case '"':
|
||||||
|
if (cp == '"') {
|
||||||
|
quote = 0;
|
||||||
|
next = Transition.FROM_WEAK;
|
||||||
|
} else if (cp == '\\') {
|
||||||
|
// Note we don't make the distinction between special chars like " or \ from other chars
|
||||||
|
// that are supposed to not escaped (i.e "\a" is \a and "\$" is $)
|
||||||
|
// this interpretation is not done by termd
|
||||||
|
next = Transition.TO_ESC;
|
||||||
|
} else {
|
||||||
|
next = Transition.CODE_POINT;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
throw new AssertionError();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
this.transition = next;
|
||||||
|
}
|
||||||
|
|
||||||
|
private enum Transition {
|
||||||
|
|
||||||
|
TO_STRONG, TO_WEAK, FROM_STRONG, FROM_WEAK, TO_ESC, CODE_POINT, FROM_ESC;
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
175
src/main/java/com/lz/common/cli/impl/CliTokenImpl.java
Normal file
175
src/main/java/com/lz/common/cli/impl/CliTokenImpl.java
Normal file
@ -0,0 +1,175 @@
|
|||||||
|
package com.lz.common.cli.impl;
|
||||||
|
|
||||||
|
|
||||||
|
import com.lz.common.cli.CliToken;
|
||||||
|
import com.lz.common.cli.LineStatus;
|
||||||
|
|
||||||
|
import java.util.LinkedList;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author <a href="mailto:julien@julienviet.com">Julien Viet</a>
|
||||||
|
*/
|
||||||
|
public class CliTokenImpl implements CliToken {
|
||||||
|
|
||||||
|
final boolean text;
|
||||||
|
final String raw;
|
||||||
|
final String value;
|
||||||
|
|
||||||
|
public CliTokenImpl(boolean text, String value) {
|
||||||
|
this(text, value, value);
|
||||||
|
}
|
||||||
|
|
||||||
|
public CliTokenImpl(boolean text, String raw, String value) {
|
||||||
|
this.text = text;
|
||||||
|
this.raw = raw;
|
||||||
|
this.value = value;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isText() {
|
||||||
|
return text;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isBlank() {
|
||||||
|
return !text;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String raw() {
|
||||||
|
return raw;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String value() {
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int hashCode() {
|
||||||
|
return value.hashCode();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean equals(Object obj) {
|
||||||
|
if (obj == this) {
|
||||||
|
return true;
|
||||||
|
} else if (obj instanceof CliTokenImpl) {
|
||||||
|
CliTokenImpl that = (CliTokenImpl) obj;
|
||||||
|
return text == that.text && value.equals(that.value);
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String toString() {
|
||||||
|
return "CliToken[text=" + text + ",value=" + value + "]";
|
||||||
|
}
|
||||||
|
|
||||||
|
public static List<CliToken> tokenize(String s) {
|
||||||
|
|
||||||
|
List<CliToken> tokens = new LinkedList<CliToken>();
|
||||||
|
|
||||||
|
tokenize(s, 0, tokens);
|
||||||
|
return tokens;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void tokenize(String s, int index, List<CliToken> builder) {
|
||||||
|
while (index < s.length()) {
|
||||||
|
char c = s.charAt(index);
|
||||||
|
switch (c) {
|
||||||
|
case ' ':
|
||||||
|
case '\t':
|
||||||
|
index = blankToken(s, index, builder);
|
||||||
|
break;
|
||||||
|
case '(':
|
||||||
|
index = khToken(s, index, builder);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
index = textToken(s, index, builder);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Todo use code points and not chars
|
||||||
|
private static int textToken(String s, int index, List<CliToken> builder) {
|
||||||
|
LineStatus quoter = new LineStatus();
|
||||||
|
int from = index;
|
||||||
|
StringBuilder value = new StringBuilder();
|
||||||
|
while (index < s.length()) {
|
||||||
|
char c = s.charAt(index);
|
||||||
|
if (isLeftKuo(c)) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
quoter.accept(c);
|
||||||
|
if (!quoter.isQuoted() && !quoter.isEscaped() && isBlank(c)) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (quoter.isCodePoint()) {
|
||||||
|
if (quoter.isEscaped() && quoter.isWeaklyQuoted() && c != '"') {
|
||||||
|
value.append('\\');
|
||||||
|
}
|
||||||
|
value.append(c);
|
||||||
|
}
|
||||||
|
index++;
|
||||||
|
}
|
||||||
|
builder.add(new CliTokenImpl(true, s.substring(from, index), value.toString()));
|
||||||
|
return index;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static int blankToken(String s, int index, List<CliToken> builder) {
|
||||||
|
int from = index;
|
||||||
|
while (index < s.length() && isBlank(s.charAt(index))) {
|
||||||
|
index++;
|
||||||
|
}
|
||||||
|
// builder.add(new CliTokenImpl(false, s.substring(from, index)));
|
||||||
|
return index;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
private static int khToken(String s, int index, List<CliToken> builder) {
|
||||||
|
int from = index;
|
||||||
|
int kuo = 0;
|
||||||
|
while (index < s.length()) {
|
||||||
|
index++;
|
||||||
|
char c = s.charAt(index);
|
||||||
|
try {
|
||||||
|
//Thread.sleep(100);
|
||||||
|
} catch (Exception e) {
|
||||||
|
e.printStackTrace();
|
||||||
|
}
|
||||||
|
if (isLeftKuo(c)) {
|
||||||
|
kuo++;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (isRightKuo(c)) {
|
||||||
|
if (kuo == 0) {
|
||||||
|
index++;
|
||||||
|
break;
|
||||||
|
} else {
|
||||||
|
kuo--;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
builder.add(new CliTokenImpl(false, s.substring(from, index)));
|
||||||
|
return index;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
private static boolean isBlank(char c) {
|
||||||
|
return c == ' ' || c == '\t';
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
private static boolean isLeftKuo(char c) {
|
||||||
|
return c == '(';
|
||||||
|
}
|
||||||
|
|
||||||
|
private static boolean isRightKuo(char c) {
|
||||||
|
return c == ')';
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
@ -8,6 +8,7 @@
|
|||||||
|
|
||||||
package com.lz.common.utils;
|
package com.lz.common.utils;
|
||||||
|
|
||||||
|
import com.lz.modules.app.utils.t.Tuple;
|
||||||
import org.apache.http.HttpStatus;
|
import org.apache.http.HttpStatus;
|
||||||
|
|
||||||
import java.util.HashMap;
|
import java.util.HashMap;
|
||||||
@ -23,6 +24,8 @@ public class R extends HashMap<String, Object> {
|
|||||||
|
|
||||||
private boolean isSuccess;
|
private boolean isSuccess;
|
||||||
|
|
||||||
|
private Tuple tuple;
|
||||||
|
|
||||||
public R() {
|
public R() {
|
||||||
put("code", 200);
|
put("code", 200);
|
||||||
put("msg", "success");
|
put("msg", "success");
|
||||||
@ -45,6 +48,19 @@ public class R extends HashMap<String, Object> {
|
|||||||
return r;
|
return r;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
public R putObj(Tuple tuple) {
|
||||||
|
this.tuple = tuple;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static R ok(int code) {
|
||||||
|
R r = new R();
|
||||||
|
r.put("code", code);
|
||||||
|
return r;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
public static R ok(String msg) {
|
public static R ok(String msg) {
|
||||||
R r = new R();
|
R r = new R();
|
||||||
r.put("msg", msg);
|
r.put("msg", msg);
|
||||||
|
|||||||
60
src/main/java/com/lz/common/utils/TaskCommand.java
Normal file
60
src/main/java/com/lz/common/utils/TaskCommand.java
Normal file
@ -0,0 +1,60 @@
|
|||||||
|
package com.lz.common.utils;
|
||||||
|
|
||||||
|
|
||||||
|
import com.lz.common.cli.CliToken;
|
||||||
|
import com.lz.common.cli.CliTokens;
|
||||||
|
import com.lz.modules.app.utils.t.Tuple;
|
||||||
|
import org.junit.Test;
|
||||||
|
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.Arrays;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
public class TaskCommand {
|
||||||
|
|
||||||
|
public static final String add = "add";
|
||||||
|
public static final String update = "update";
|
||||||
|
public static final String list = "list";
|
||||||
|
public static final String task = "task";
|
||||||
|
public static final String record = "record";
|
||||||
|
|
||||||
|
public static List<String> commandCommands = new ArrayList<String>(Arrays.asList(new String[]{"add","update","list"}));
|
||||||
|
|
||||||
|
public static List<String> optTargets = new ArrayList<String>(Arrays.asList(new String[]{"task", "record"}));
|
||||||
|
|
||||||
|
|
||||||
|
public static Tuple parse(String command) {
|
||||||
|
String[] tokens = getTokens(command);
|
||||||
|
if (commandCommands.contains(tokens[0])) {
|
||||||
|
return new Tuple(488, "命令的第一个参数必需是 add or update or list");
|
||||||
|
}
|
||||||
|
if (optTargets.contains(tokens[1])) {
|
||||||
|
return new Tuple(488, "命令的第二个参数必需是 task or record");
|
||||||
|
}
|
||||||
|
if (task.equals(tokens[1])) { //表示对 task 增删改查
|
||||||
|
|
||||||
|
} else if (record.equals(tokens[1])) { //表示对 list 增删改查
|
||||||
|
|
||||||
|
}
|
||||||
|
return new Tuple();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
public static String [] getTokens(String command) {
|
||||||
|
List<String> tokenList = new ArrayList<>();
|
||||||
|
List<CliToken> tokens = CliTokens.tokenize(command);
|
||||||
|
for (int i = 0; i < tokens.size(); i++) {
|
||||||
|
tokenList.add(tokens.get(i).value());
|
||||||
|
}
|
||||||
|
return tokenList.toArray(new String[tokenList.size()]);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void test11() {
|
||||||
|
String ln = " abcdef b csssss dddddd -c \"b ccc ' bb a , bb x b\" a ";
|
||||||
|
List<CliToken> tokens = CliTokens.tokenize(ln);
|
||||||
|
for (int i = 0; i < tokens.size(); i++) {
|
||||||
|
System.out.println(tokens.get(i).value());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -43,6 +43,9 @@ public class ThirdTaskController {
|
|||||||
}
|
}
|
||||||
|
|
||||||
log.info("command:" + command);
|
log.info("command:" + command);
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
List<List<String>> data = new ArrayList<>();
|
List<List<String>> data = new ArrayList<>();
|
||||||
for (int i = 0; i < 5; i++) {
|
for (int i = 0; i < 5; i++) {
|
||||||
List<String> a = new ArrayList<>();
|
List<String> a = new ArrayList<>();
|
||||||
@ -52,6 +55,10 @@ public class ThirdTaskController {
|
|||||||
data.add(a);
|
data.add(a);
|
||||||
}
|
}
|
||||||
List<String> header = new ArrayList<>(Arrays.asList(new String[]{"用户名", "密码", "哈哈", "双", "你是"}));
|
List<String> header = new ArrayList<>(Arrays.asList(new String[]{"用户名", "密码", "哈哈", "双", "你是"}));
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
return R.ok().put("header", header).put("data", data);
|
return R.ok().put("header", header).put("data", data);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
Loading…
x
Reference in New Issue
Block a user