diff --git a/DrissionPage/_elements/chromium_element.py b/DrissionPage/_elements/chromium_element.py index 248b628..ef0cb83 100644 --- a/DrissionPage/_elements/chromium_element.py +++ b/DrissionPage/_elements/chromium_element.py @@ -93,6 +93,13 @@ class ChromiumElement(DrissionElement): def __eq__(self, other): return self._backend_id == getattr(other, '_backend_id', None) + def __getattr__(self, item): + """获取元素属性 + :param item: 属性名 + :return: 属性值 + """ + return self.attr(item) or self.property(item) + @property def tag(self): """返回元素tag""" diff --git a/DrissionPage/_elements/chromium_element.pyi b/DrissionPage/_elements/chromium_element.pyi index 730ebb1..bf62dfb 100644 --- a/DrissionPage/_elements/chromium_element.pyi +++ b/DrissionPage/_elements/chromium_element.pyi @@ -56,6 +56,8 @@ class ChromiumElement(DrissionElement): def __eq__(self, other: ChromiumElement) -> bool: ... + def __getattr__(self, item: str) -> str: ... + @property def tag(self) -> str: ... diff --git a/DrissionPage/_elements/session_element.py b/DrissionPage/_elements/session_element.py index 07c2a87..4fe9c77 100644 --- a/DrissionPage/_elements/session_element.py +++ b/DrissionPage/_elements/session_element.py @@ -50,6 +50,13 @@ class SessionElement(DrissionElement): def __eq__(self, other): return self.xpath == getattr(other, 'xpath', None) + def __getattr__(self, item): + """获取元素属性 + :param item: 属性名 + :return: 属性值 + """ + return self.attr(item) + @property def tag(self): """返回元素类型""" diff --git a/DrissionPage/_elements/session_element.pyi b/DrissionPage/_elements/session_element.pyi index f6e4e00..74034a9 100644 --- a/DrissionPage/_elements/session_element.pyi +++ b/DrissionPage/_elements/session_element.pyi @@ -35,6 +35,8 @@ class SessionElement(DrissionElement): def __eq__(self, other: SessionElement) -> bool: ... + def __getattr__(self, item: str) -> str: ... + @property def tag(self) -> str: ... diff --git a/DrissionPage/_pages/chromium_base.py b/DrissionPage/_pages/chromium_base.py index 1c73d00..153ba75 100644 --- a/DrissionPage/_pages/chromium_base.py +++ b/DrissionPage/_pages/chromium_base.py @@ -156,7 +156,7 @@ class ChromiumBase(BasePage): if self._is_reading: return self._is_reading = True - timeout = timeout if timeout >= .5 else .5 + timeout = max(timeout, 2) end_time = perf_counter() + timeout while perf_counter() < end_time: try: diff --git a/DrissionPage/_pages/chromium_frame.py b/DrissionPage/_pages/chromium_frame.py index fef99d0..f2f08de 100644 --- a/DrissionPage/_pages/chromium_frame.py +++ b/DrissionPage/_pages/chromium_frame.py @@ -164,7 +164,7 @@ class ChromiumFrame(ChromiumBase): self.doc_ele = ChromiumElement(self._target_page, backend_id=node['contentDocument']['backendNodeId']) else: - timeout = timeout if timeout >= .5 else .5 + timeout = max(timeout, 2) b_id = self.run_cdp('DOM.getDocument', _timeout=timeout)['root']['backendNodeId'] self.doc_ele = ChromiumElement(self, backend_id=b_id) @@ -176,7 +176,6 @@ class ChromiumFrame(ChromiumBase): return True except: - raise return False finally: diff --git a/java/pom.xml b/java/pom.xml deleted file mode 100644 index 17ea12c..0000000 --- a/java/pom.xml +++ /dev/null @@ -1,154 +0,0 @@ - - 4.0.0 - com.ll - DrissonPage - 0.0.1 - Magic - DrissonPage - http://maven.apache.org - - 11 - 11 - - - - - org.apache.commons - commons-collections4 - 4.4 - - - org.projectlombok - lombok - 1.18.30 - - - - org.apache.commons - commons-text - 1.11.0 - - - - org.ini4j - ini4j - 0.5.4 - - - - com.alibaba - fastjson - 1.2.83 - - - - net.java.dev.jna - jna - 5.10.0 - - - net.java.dev.jna - jna-platform - 5.10.0 - - - - - - - - - - - - - com.squareup.okhttp3 - okhttp - 4.10.0 - - - - com.squareup.okio - okio-jvm - 3.4.0 - - - - org.apache.httpcomponents - httpasyncclient - 4.1.5 - - - - commons-codec - commons-codec - 1.16.0 - - - org.apache.maven - maven-model - 3.8.1 - - - org.apache.maven - maven-core - 3.8.1 - - - - org.apache.maven.shared - maven-shared-utils - 3.3.4 - - - - commons-io - commons-io - 2.11.0 - - - - com.google.guava - guava - 32.1.2-jre - - - - org.jsoup - jsoup - 1.16.1 - - - org.jetbrains - annotations - RELEASE - compile - - - - org.apache.poi - poi - 5.1.0 - - - org.apache.poi - poi-ooxml - 5.1.0 - - - - org.apache.commons - commons-compress - 1.21 - - - - - - - diff --git a/java/src/main/java/com/ll/DataRecorder/BaseRecorder.java b/java/src/main/java/com/ll/DataRecorder/BaseRecorder.java deleted file mode 100644 index 7c0f116..0000000 --- a/java/src/main/java/com/ll/DataRecorder/BaseRecorder.java +++ /dev/null @@ -1,76 +0,0 @@ -package com.ll.DataRecorder; - -import java.nio.file.Path; - -/** - * @author 陆 - * @address click - */ -public abstract class BaseRecorder extends OriginalRecorder { - protected String encoding; - protected Object before; - protected Object after; - protected String table; - - public BaseRecorder() { - } - - public BaseRecorder(Integer cacheSize) { - super(cacheSize); - } - - public BaseRecorder(Path path) { - super(path); - } - - public BaseRecorder(Path path, Integer cacheSize) { - super(path, cacheSize); - } - - public BaseRecorder(String path) { - super(path); - } - - public BaseRecorder(String path, Integer cacheSize) { - super(path, cacheSize); - } - - /** - * @return 返回用于设置属性的对象 - */ - @Override - public BaseSetter> set() { - if (super.setter == null) super.setter = new BaseSetter<>(this); - return (BaseSetter>) super.setter; - } - - /** - * @return 返回当前before内容 - */ - public Object before() { - return this.before; - } - - - /** - * @return 返回当前before内容 - */ - public Object after() { - return this.after; - } - - /** - * @return 返回默认表名 - */ - public String table() { - return this.table; - } - - /** - * @return 返回编码格式 - */ - public String encoding() { - return this.encoding; - } - -} diff --git a/java/src/main/java/com/ll/DataRecorder/BaseSetter.java b/java/src/main/java/com/ll/DataRecorder/BaseSetter.java deleted file mode 100644 index 56d1a7d..0000000 --- a/java/src/main/java/com/ll/DataRecorder/BaseSetter.java +++ /dev/null @@ -1,155 +0,0 @@ -package com.ll.DataRecorder; - -import java.nio.charset.Charset; -import java.util.Collections; -import java.util.List; -import java.util.Map; - -/** - * @author 陆 - * @address click - */ -public class BaseSetter extends OriginalSetter { - public BaseSetter(R recorder) { - super(recorder); - } - - /** - * 设置默认表名 - * - * @param name 表名 - */ - public void table(String name) { - recorder.table = name; - } - - /** - * 设置在数据前面补充的列 - * - * @param before 列表、数组或字符串,为字符串时则补充一列 - */ - public void before(List before) { - before(before, false); - } - - /** - * 设置在数据前面补充的列 - * - * @param before 列表、数组或字符串,为字符串时则补充一列 - */ - public void before(Map, ?> before) { - before(before, false); - } - - /** - * 设置在数据前面补充的列 - * - * @param before 列表、数组或字符串,为字符串时则补充一列 - */ - public void before(String before) { - before(before, true); - } - - /** - * 设置在数据前面补充的列 - * - * @param before 列表、数组或字符串,为字符串时则补充一列 - */ - public void before(String[] before) { - before(before, false); - } - - /** - * 设置在数据前面补充的列 - * - * @param before 列表、数组或字符串,为字符串时则补充一列 - */ - private void before(Object before, boolean ignoredI) { - if (before == null || "".equals(before)) { - this.recorder.before = null; - } else if (before instanceof List) { - this.recorder.before = before; - } else if (before instanceof Map) { - this.recorder.before = before; - - } else if (before instanceof String[]) { - this.recorder.before = before; - } else { - this.recorder.before = Collections.singletonList(before); - } - } - - /** - * 设置在数据后面补充的列 - * - * @param after 列表、数组或字符串,为字符串时则补充一列 - */ - public void after(List after) { - after(after, false); - } - - /** - * 设置在数据后面补充的列 - * - * @param after 列表、数组或字符串,为字符串时则补充一列 - */ - public void after(Map, ?> after) { - after(after, false); - } - - /** - * 设置在数据后面补充的列 - * - * @param after 列表、数组或字符串,为字符串时则补充一列 - */ - public void after(String after) { - after(after, true); - } - - /** - * 设置在数据后面补充的列 - * - * @param after 列表、数组或字符串,为字符串时则补充一列 - */ - public void after(String[] after) { - after(after, false); - } - - /** - * 设置在数据后面补充的列 - * - * @param after 列表、数组或字符串,为字符串时则补充一列 - */ - private void after(Object after, boolean ignoredI) { - if (after == null || "".equals(after)) { - this.recorder.after = null; - } else if (after instanceof List) { - this.recorder.after = after; - } else if (after instanceof Map) { - this.recorder.after = after; - - } else if (after instanceof String[]) { - this.recorder.after = after; - } else { - this.recorder.after = Collections.singletonList(after); - } - } - - /** - * 设置编码 - * - * @param encoding 编码格式 - */ - public void encoding(String encoding) { - this.recorder.encoding = Charset.forName(encoding).name(); - } - - /** - * 设置编码 - * - * @param encoding 编码格式 - */ - public void encoding(Charset encoding) { - this.recorder.encoding = encoding.name(); - } -} diff --git a/java/src/main/java/com/ll/DataRecorder/ByteRecorder.java b/java/src/main/java/com/ll/DataRecorder/ByteRecorder.java deleted file mode 100644 index 042501f..0000000 --- a/java/src/main/java/com/ll/DataRecorder/ByteRecorder.java +++ /dev/null @@ -1,150 +0,0 @@ -package com.ll.DataRecorder; - -import lombok.AllArgsConstructor; -import lombok.Getter; - -import java.io.IOException; -import java.nio.ByteBuffer; -import java.nio.channels.SeekableByteChannel; -import java.nio.file.Files; -import java.nio.file.Path; -import java.nio.file.Paths; -import java.nio.file.StandardOpenOption; -import java.util.List; - -/** - * @author 陆 - * @address click - */ -public class ByteRecorder extends OriginalRecorder { - private static final long[] END = {0, 2}; - protected List data; - - /** - * 用于记录字节数据的工具 - */ - public ByteRecorder() { - this(""); - } - - /** - * 用于记录字节数据的工具 - * - * @param path 保存的文件路径 - */ - public ByteRecorder(Path path) { - this(path, null); - } - - /** - * 用于记录字节数据的工具 - * - * @param cacheSize 每接收多少条记录写入文件,0为不自动写入 - */ - public ByteRecorder(Integer cacheSize) { - this("", null); - } - - /** - * 用于记录字节数据的工具 - * - * @param path 保存的文件路径 - * @param cacheSize 每接收多少条记录写入文件,0为不自动写入 - */ - public ByteRecorder(Path path, Integer cacheSize) { - super(path == null ? null : path.toAbsolutePath().toString(), cacheSize); - } - - /** - * 用于记录字节数据的工具 - * - * @param path 保存的文件路径 - */ - public ByteRecorder(String path) { - super(path, null); - } - - /** - * 用于记录字节数据的工具 - * - * @param path 保存的文件路径 - * @param cacheSize 每接收多少条记录写入文件,0为不自动写入 - */ - public ByteRecorder(String path, Integer cacheSize) { - super("".equals(path) ? null : path, cacheSize); - } - - /** - * @param data 类型只能为byte[] - */ - @Override - public void addData(Object data) { - if (data instanceof byte[]) addData((byte[]) data, null); - else throw new IllegalArgumentException("data类型只能为byte[]为了兼容"); - } - - - /** - * 添加一段二进制数据 - * - * @param data bytes类型数据 - * @param seek 在文件中的位置,None表示最后 - */ - public void addData(byte[] data, Long seek) { - while (this.pauseAdd) { //等待其它线程写入结束 - try { - Thread.sleep(100); - } catch (InterruptedException e) { - throw new RuntimeException(e); - } - } - if (seek != null && seek < 0) throw new IllegalArgumentException("seek参数只能接受null或大于等于0的整数。"); - this.data.add(new ByteData(data, seek)); - this.dataCount++; - if (0 < this.cacheSize() && this.cacheSize() <= this.dataCount) this.record(); - - } - - /** - * @return 返回当前保存在缓存的数据 - */ - public List data() { - return this.data; - } - - /** - * 记录数据到文件 - */ - protected void _record() { - Path filePath = Paths.get(path); - if (!Files.exists(filePath)) { - try { - Files.createFile(filePath); - } catch (IOException e) { - throw new RuntimeException(e); - } - } - - try (SeekableByteChannel fileChannel = Files.newByteChannel(filePath, StandardOpenOption.WRITE, StandardOpenOption.READ)) { - long[] previous = null; - for (ByteData entry : data) { - - long[] loc = (entry.seek == null ? ByteRecorder.END : new long[]{entry.seek, 0}); - if (!(previous != null && previous[0] == loc[0] && previous[1] == loc[1] && ByteRecorder.END[0] == loc[0] && ByteRecorder.END[1] == loc[1])) { - fileChannel.position(loc[0] + loc[1]); - previous = loc; - } - fileChannel.write(ByteBuffer.wrap(entry.data)); - } - } catch (IOException e) { - throw new RuntimeException(e); - } - } - - @Getter - @AllArgsConstructor - public static class ByteData { - private byte[] data; - private Long seek; - } -} diff --git a/java/src/main/java/com/ll/DataRecorder/DBRecorder.java b/java/src/main/java/com/ll/DataRecorder/DBRecorder.java deleted file mode 100644 index 781d33c..0000000 --- a/java/src/main/java/com/ll/DataRecorder/DBRecorder.java +++ /dev/null @@ -1,20 +0,0 @@ -package com.ll.DataRecorder; - -import javax.naming.NoPermissionException; - -/** - * 用于存储数据到sqlite的工具 - * @author 陆 - * @address click - */ -public class DBRecorder extends BaseRecorder { - @Override - public void addData(Object data) { - - } - - @Override - protected void _record() throws NoPermissionException { - - } -} diff --git a/java/src/main/java/com/ll/DataRecorder/OriginalRecorder.java b/java/src/main/java/com/ll/DataRecorder/OriginalRecorder.java deleted file mode 100644 index 62a089d..0000000 --- a/java/src/main/java/com/ll/DataRecorder/OriginalRecorder.java +++ /dev/null @@ -1,236 +0,0 @@ -package com.ll.DataRecorder; - -import javax.naming.NoPermissionException; -import java.io.IOException; -import java.nio.file.Files; -import java.nio.file.Path; -import java.nio.file.Paths; -import java.nio.file.StandardCopyOption; -import java.util.List; -import java.util.concurrent.locks.Lock; -import java.util.concurrent.locks.ReentrantLock; - -/** - * 记录器的基类 - * - * @author 陆 - * @address click - */ -public abstract class OriginalRecorder { - protected int cache; - protected String path; - protected String type; - protected List data; - protected Lock lock = new ReentrantLock();//线程锁 - protected boolean pauseAdd = false; - protected boolean pauseWrite = false; - public boolean showMsg = true; - protected OriginalSetter> setter; - protected int dataCount = 0; - - /** - * - */ - public OriginalRecorder() { - this(""); - } - - /** - * @param cacheSize 每接收多少条记录写入文件,0为不自动写入 - */ - public OriginalRecorder(Integer cacheSize) { - this("", cacheSize); - } - - /** - * @param path 保存的文件路径 - */ - public OriginalRecorder(Path path) { - this(path.toString(), null); - } - - /** - * @param path 保存的文件路径 - * @param cacheSize 每接收多少条记录写入文件,0为不自动写入 - */ - public OriginalRecorder(Path path, Integer cacheSize) { - this(path.toString(), cacheSize); - } - - /** - * @param path 保存的文件路径 - */ - public OriginalRecorder(String path) { - this(path, null); - } - - /** - * @param path 保存的文件路径 - * @param cacheSize 每接收多少条记录写入文件,0为不自动写入 - */ - public OriginalRecorder(String path, Integer cacheSize) { - this.set().path(path); - this.cache = cacheSize != null ? cacheSize : 1000; - } - - /** - * @return 返回用于设置属性的对象 - */ - public OriginalSetter> set() { - if (this.setter == null) this.setter = new OriginalSetter<>(this); - return this.setter; - } - - /** - * @return 返回缓存大小 - */ - public int cacheSize() { - return this.cache; - } - - /** - * @return 返回文件路径 - */ - public String path() { - return this.path; - } - - /** - * @return 返回文件类型 - */ - public String type() { - return this.type; - } - - /** - * @return 返回当前保存在缓存的数据 - */ - public Object data() { - return this.data; - } - - /** - * 记录数据,可保存到新文件 - * - * @return 文件路径 - */ - public String record() { - return record(""); - } - - /** - * 记录数据,可保存到新文件 - * - * @param newPath 文件另存为的路径,会保存新文件 - * @return 文件路径 - */ - - public String record(Path newPath) { - return record(newPath.toString()); - } - - - /** - * 记录数据,可保存到新文件 - * - * @param newPath 文件另存为的路径,会保存新文件 - * @return 文件路径 - * @throws IOException 读写文件时可能发生IOException - */ - public String record(String newPath) { - if ("".equals(newPath)) newPath = null; - // 具体功能由_record()实现,本方法实现自动重试及另存文件功能 - String originalPath = path; - String returnPath = path; - if (newPath != null && !newPath.isEmpty()) { - newPath = Tools.getUsablePath(newPath).toString(); - returnPath = path = newPath; - - Path originalFilePath = Paths.get(originalPath); - if (Files.exists(originalFilePath)) { - Path newPathObject = Paths.get(newPath); - try { - Files.copy(originalFilePath, newPathObject, StandardCopyOption.REPLACE_EXISTING); - } catch (IOException e) { - throw new RuntimeException(e); - } - } - } - - if (!data.isEmpty()) { - return returnPath; - } - - if (path == null || path.isEmpty()) { - throw new IllegalArgumentException("保存路径为空。"); - } - - lock.lock(); - try { - pauseAdd = true; // 写入文件前暂缓接收数据 - if (showMsg) { - System.out.println(path + " 开始写入文件,切勿关闭进程。"); - } - - try { - Files.createDirectories(Paths.get(path).getParent()); - } catch (IOException e) { - throw new RuntimeException(e); - } - while (true) { - try { - while (pauseWrite) { // 等待其它线程写入结束 - Thread.sleep(100); - } - pauseWrite = true; - this._record(); - break; - - } catch (NoPermissionException e) { - - } catch (Exception e) { - try { - Files.write(Paths.get("failed_data.txt"), (data.toString() + "\n").getBytes()); - System.out.println("保存失败的数据已保存到failed_data.txt。"); - } catch (IOException ioException) { - throw e; - } - throw e; - } finally { - pauseWrite = false; - } - - Thread.sleep(300); - } - - if (newPath != null && !newPath.isEmpty()) { - path = originalPath; - } - - if (showMsg) { - System.out.println(path + " 写入文件结束。"); - } - clear(); - dataCount = 0; - pauseAdd = false; - - } catch (InterruptedException e) { - throw new RuntimeException(e); - } finally { - lock.unlock(); - } - - return returnPath; - } - - /** - * 清空缓存中的数据 - */ - public void clear() { - if (this.data != null) this.data.clear(); - } - - public abstract void addData(Object data); - - protected abstract void _record() throws NoPermissionException; -} diff --git a/java/src/main/java/com/ll/DataRecorder/OriginalSetter.java b/java/src/main/java/com/ll/DataRecorder/OriginalSetter.java deleted file mode 100644 index 8661fc5..0000000 --- a/java/src/main/java/com/ll/DataRecorder/OriginalSetter.java +++ /dev/null @@ -1,58 +0,0 @@ -package com.ll.DataRecorder; - -import lombok.AllArgsConstructor; - -import java.nio.file.Path; -import java.util.ArrayList; - -/** - * @author 陆 - * @address click - */ -@AllArgsConstructor -public class OriginalSetter { - protected final R recorder; - - - /** - * 设置缓存大小 - * - * @param size 缓存大小 - */ - public void cacheSize(int size) { - if (size < 0) return; - this.recorder.cache = size; - } - - /** - * 设置文件路径 - * - * @param path 文件路径 - */ - public void path(String path) { - if (this.recorder.path() != null) this.recorder.record(); - this.recorder.path = path; - this.recorder.data = new ArrayList<>(); - } - - /** - * 设置文件路径 - * - * @param path 文件路径 - */ - public void path(Path path) { - if (this.recorder.path() != null) this.recorder.record(); - this.recorder.path = path.toString(); - this.recorder.data = new ArrayList<>(); - } - - /** - * 设置是否显示运行信息 - * - * @param onOff 开关 - */ - - public void showMsg(boolean onOff) { - this.recorder.showMsg = onOff; - } -} diff --git a/java/src/main/java/com/ll/DataRecorder/Recorder.java b/java/src/main/java/com/ll/DataRecorder/Recorder.java deleted file mode 100644 index 8a3b3df..0000000 --- a/java/src/main/java/com/ll/DataRecorder/Recorder.java +++ /dev/null @@ -1,269 +0,0 @@ -package com.ll.DataRecorder; - -import java.nio.file.Path; -import java.util.ArrayList; -import java.util.Collections; -import java.util.List; -import java.util.Map; - -/** - * @author 陆 - * @address click - */ -public class Recorder extends BaseRecorder { - - private String delimiter = ","; - private String quoteChar = "\""; - private boolean followStyles = false; - private Float colHeight = null; - private String style = null; - private boolean fitHead = false; - - // 其他属性和方法的声明 - public Recorder(String path) { - this(path, null); - } - - public Recorder(Path path) { - this(path.toString(), null); - } - - public Recorder(String path, Integer cacheSize) { - super(path, cacheSize); - } - - // 其他方法和属性的具体实现 - - public RecorderSetter set() { - if (setter == null) { - setter = new RecorderSetter(this); - } - return (RecorderSetter) setter; - } - - @Override - public void addData(Object data) { - this.addData(data, null); - } - - /** - * @return 返回csv文件分隔符 - */ - public String delimiter() { - return delimiter; - } - - /** - * @return 返回csv文件引用符 - */ - public String quoteChar() { - return quoteChar; - } - - public void addData(Object data, String table) { - while (this.pauseAdd) { - try { - Thread.sleep(100); - } catch (InterruptedException e) { - e.printStackTrace(); - } - } - - if (!(data instanceof List) && !(data instanceof Map)) { - data = Collections.singletonList(data); - } - - if (data instanceof List> && ((List>) data).isEmpty()) { - data = new ArrayList<>(); - this.dataCount++; - } - - if (!"xlsx".equals(type)) { - if (data instanceof List>) { - this.data.addAll((List>) data); - } - } else { - if (table == null) { - table = this.table; - } else if (table.equals("false")) { - table = null; - } - -// this.data.add(table, k -> new ArrayList<>()).addAll((List>) data); - } - - if (0 < this.cache && cache <= dataCount) { - record(); - } - } - - protected void _record() { -// if ("csv".equals(type)) { -// toCsv(); -// } else if ("xlsx".equals(type)) { -// toXlsx(); -// } else if ("json".equals(type)) { -// toJson(); -// } else if ("txt".equals(type)) { -// toTxt(); -// } - } - -// protected void toXlsx() { -// Path filePath = Paths.get(path); -// boolean newFile = filePath.toFile().exists(); -// Workbook wb; -// if (newFile) { -// try { -// wb = WorkbookFactory.create(filePath.toFile()); -// } catch (IOException e) { -// e.printStackTrace(); -// return; -// } -// } else { -// wb = new XSSFWorkbook(); -// } -// -// List tables = new ArrayList<>(); -// data.forEach((table) -> { -// boolean newSheet = false; -// Sheet sheet; -// if (table == null) { -// sheet = wb.getSheetAt(0); -// } else if (tables.contains(table)) { -// sheet = wb.getSheet(table); -// } else if (newFile) { -// sheet = wb.getSheetAt(0); -// tables.remove(sheet.getSheetName()); -// sheet.getWorkbook().setSheetName(sheet.getWorkbook().getSheetIndex(sheet), table); -// } else { -// sheet = wb.createSheet(table); -// tables.add(table); -// newSheet = true; -// } -// -// if (newFile || newSheet) { -// List title = getTitle(data, before(), after()); -// if (title != null) { -// Row titleRow = sheet.createRow(sheet.getPhysicalNumberOfRows()); -// title.forEach(t -> titleRow.createCell(titleRow.getPhysicalNumberOfCells()).setCellValue(t)); -// } -// } -// -// Float colHeight = null; -// List rowStyles = null; -// if (newFile || newSheet) { -// if (this.colHeight != null || followStyles || style != null || _data.size() > 1) { -// wb.getCreationHelper().createFormulaEvaluator().evaluateAll(); -// } -// } -// -// if (newFile || newSheet) { -// if (this.colHeight != null || followStyles) { -// int lastRowNum = sheet.getLastRowNum(); -// Row lastRow = sheet.getRow(lastRowNum); -// if (lastRow != null) { -// colHeight = lastRow.getHeightInPoints(); -// if (followStyles) { -// rowStyles = new ArrayList<>(); -// for (Cell cell : lastRow) { -// rowStyles.add(new CellBase(cell.getCellStyle())); -// } -// } -// } -// } -// } -// -// if (newFile || newSheet) { -// if (fitHead && _head.get(sheet.getSheetName()) == null) { -// _head.put(sheet.getSheetName(), getTitle(data.get(0), _before, _after)); -// } -// } -// -// if (fitHead && _head.get(sheet.getSheetName()) != null) { -// data.forEach(d -> { -// if (d instanceof Map) { -// d = ((Map, ?>) d).values(); -// } -// Row row = sheet.createRow(sheet.getPhysicalNumberOfRows()); -// _head.get(sheet.getSheetName()).forEach(h -> row.createCell(row.getPhysicalNumberOfCells()).setCellValue(processContent(((Map, ?>) d).get(h)))); -// setStyle(colHeight, rowStyles, row); -// }); -// } else { -// data.forEach(d -> { -// if (d instanceof Map) { -// d = ((Map, ?>) d).values(); -// } -// Row row = sheet.createRow(sheet.getPhysicalNumberOfRows()); -// ((List>) d).forEach(value -> row.createCell(row.getPhysicalNumberOfCells()).setCellValue(processContent(value))); -// setStyle(colHeight, rowStyles, row); -// }); -// } -// }); -// -// try { -// wb.write(filePath.toFile()); -// wb.close(); -// } catch (IOException e) { -// e.printStackTrace(); -// } -// } -// -// protected void setStyle(Float colHeight, List rowStyles, Row row) { -// if (colHeight != null) { -// row.setHeightInPoints(colHeight); -// } -// -// if (rowStyles != null) { -// for (int i = 0; i < row.getPhysicalNumberOfCells(); i++) { -// Cell cell = row.getCell(i); -// CellStyleCopier styleCopier = rowStyles.get(i); -// styleCopier.setToCell(cell); -// } -// } else if (style != null) { -// for (Cell cell : row) { -// setStyle(style, cell); -// } -// } -// } -// -// protected void setStyle(String style, Table.Cell cell) { -// // 实现设置样式的逻辑 -// } -// -// protected List getTitle(Object data, Object before, Object after) { -// if (data instanceof List) { -// return null; -// } -// -// List returnList = new ArrayList<>(); -// List beforeList = getList(before); -// List afterList = getList(after); -// -// for (Object obj : List.of(beforeList, data, afterList)) { -// if (obj instanceof Map) { -// returnList.addAll(((Map, ?>) obj).keySet()); -// } else if (obj == null) { -// // Do nothing -// } else if (obj instanceof List) { -// ((List>) obj).forEach(o -> returnList.add("")); -// } else { -// returnList.add(""); -// } -// } -// -// return returnList; -// } -// -// protected List getList(Object obj) { -// if (obj instanceof List) { -// return (List) obj; -// } else if (obj instanceof Map) { -// return new ArrayList<>(((Map, ?>) obj).keySet()); -// } else { -// return List.of(""); -// } -// } - - // 其他方法和属性的具体实现 -} \ No newline at end of file diff --git a/java/src/main/java/com/ll/DataRecorder/RecorderSetter.java b/java/src/main/java/com/ll/DataRecorder/RecorderSetter.java deleted file mode 100644 index b6e5d30..0000000 --- a/java/src/main/java/com/ll/DataRecorder/RecorderSetter.java +++ /dev/null @@ -1,11 +0,0 @@ -package com.ll.DataRecorder; - -/** - * @author 陆 - * @address click - */ -public class RecorderSetter extends SheetLikeSetter { - public RecorderSetter(R recorder) { - super(recorder); - } -} diff --git a/java/src/main/java/com/ll/DataRecorder/SheetLikeSetter.java b/java/src/main/java/com/ll/DataRecorder/SheetLikeSetter.java deleted file mode 100644 index 558ccb1..0000000 --- a/java/src/main/java/com/ll/DataRecorder/SheetLikeSetter.java +++ /dev/null @@ -1,12 +0,0 @@ -package com.ll.DataRecorder; - -/** - * @author 陆 - * @address click - */ -public class SheetLikeSetter extends BaseSetter { - public SheetLikeSetter(R recorder) { - super(recorder); - } - -} diff --git a/java/src/main/java/com/ll/DataRecorder/Tools.java b/java/src/main/java/com/ll/DataRecorder/Tools.java deleted file mode 100644 index 82e49c3..0000000 --- a/java/src/main/java/com/ll/DataRecorder/Tools.java +++ /dev/null @@ -1,104 +0,0 @@ -package com.ll.DataRecorder; - -import java.nio.file.Files; -import java.nio.file.Path; -import java.nio.file.Paths; -import java.util.regex.Matcher; -import java.util.regex.Pattern; - -/** - * @author 陆 - * @address click - */ -public class Tools { - - - /** - * 检查文件或文件夹是否有重名,并返回可以使用的路径 - * @param path 文件或文件夹路径 - * @return 可用的路径,Path对象 - */ - public static Path getUsablePath(String path) { - return getUsablePath(path, true, true); - } - - /** - * 检查文件或文件夹是否有重名,并返回可以使用的路径 - * - * @param path 文件或文件夹路径 - * @param isFile 目标是文件还是文件夹 - * @param createParents 是否创建目标路径 - * @return 可用的路径,Path对象 - */ - public static Path getUsablePath(String path, boolean isFile, boolean createParents) { - Path filePath = Paths.get(path).toAbsolutePath(); - Path parent = filePath.getParent(); - if (createParents) parent.toFile().mkdirs(); - String name = makeValidName(filePath.getFileName().toString()); - int num; - String srcName; - boolean firstTime = true; - - while (Files.exists(filePath) && (Files.isRegularFile(filePath) == isFile)) { - Matcher matcher = Pattern.compile("(.*)_(\\d+)$").matcher(name); - if (!matcher.find() || (matcher.find() && firstTime)) { - srcName = name; - num = 1; - } else { - srcName = matcher.group(1); - num = Integer.parseInt(matcher.group(2)) + 1; - } - name = srcName + "_" + num; - filePath = parent.resolve(name); - firstTime = false; - } - - return filePath; - } - - /** - * 获取有效的文件名 - * - * @param fullName 文件名 - * @return 可用的文件名 - */ - public static String makeValidName(String fullName) { - fullName = fullName.trim(); - - String name; - String ext; - int extLong; - - Matcher matcher = Pattern.compile("(.*)(\\.[^.]+$)").matcher(fullName); - if (matcher.find()) { - name = matcher.group(1); - ext = matcher.group(2); - extLong = ext.length(); - } else { - name = fullName; - ext = ""; - extLong = 0; - } - - while (getLong(name) > 255 - extLong) { - name = name.substring(0, name.length() - 1); - } - - fullName = name + ext; - - return fullName.replaceAll("[<>/\\\\|:*?\\n]", ""); - } - - /** - * 返回字符串中字符个数(一个汉字是2个字符) - * - * @param txt 字符串 - * @return 字符个数 - */ - public static int getLong(String txt) { - int txtLen = txt.length(); - return (txt.getBytes().length - txtLen) / 2 + txtLen; - } - - -} diff --git a/java/src/main/java/com/ll/DownloadKit/DownloadKit.java b/java/src/main/java/com/ll/DownloadKit/DownloadKit.java deleted file mode 100644 index 474fb8a..0000000 --- a/java/src/main/java/com/ll/DownloadKit/DownloadKit.java +++ /dev/null @@ -1,1294 +0,0 @@ -package com.ll.DownloadKit; - -import com.ll.DataRecorder.Recorder; -import com.ll.DownloadKit.mission.BaseTask; -import com.ll.DownloadKit.mission.Mission; -import com.ll.DownloadKit.mission.Task; -import com.ll.DrissonPage.base.BasePage; -import com.ll.DrissonPage.config.SessionOptions; -import lombok.AllArgsConstructor; -import lombok.Getter; -import okhttp3.*; -import org.apache.commons.collections4.map.CaseInsensitiveMap; - -import java.io.IOException; -import java.math.BigDecimal; -import java.nio.file.Files; -import java.nio.file.Path; -import java.nio.file.Paths; -import java.util.*; -import java.util.concurrent.*; -import java.util.concurrent.locks.Lock; -import java.util.concurrent.locks.ReentrantLock; - -/** - * 下载器对象 - * - * @author 陆 - * @address click - */ -@Getter -public class DownloadKit { - /** - * 保存路径 - */ - protected String goalPath = "."; - /** - * 最大线程 - */ - protected int roads = 10; - private Setter setter; - protected String printMode; - protected String logMode; - protected Recorder logger; - protected Integer retry; - protected Double interval; - protected BasePage> page; - private BlockingQueue waitingList; - protected OkHttpClient session; - private int runningCount; - private int missionsNum; - private Map missions; - protected ThreadPoolExecutor threads; - protected Map> threadMap; - protected Double timeout; - private boolean stopPrinting; - private final Lock lock = new ReentrantLock();//线程锁 - protected boolean split; - protected Long blockSize; - protected String encoding; - /** - * 有同名文件名时的处理方式,可选 'skip', 'overwrite', 'rename', 'add' - */ - protected FileMode fileMode = FileMode.rename; - - /** - * 使用的Session对象,或配置对象、页面对象等 - */ - - public DownloadKit(Path goalPath, Integer roads, FileMode fileMode, BasePage> driver) { - this(goalPath.toAbsolutePath().toString(), roads, fileMode, driver); - } - - public DownloadKit(String goalPath, Integer roads, FileMode fileMode, BasePage> driver) { - init(goalPath, roads, fileMode); - this.set().driver(driver); - } - - public DownloadKit(Path goalPath, Integer roads, FileMode fileMode, OkHttpClient driver) { - this(goalPath.toAbsolutePath().toString(), roads, fileMode, driver); - } - - public DownloadKit(String goalPath, Integer roads, FileMode fileMode, OkHttpClient driver) { - init(goalPath, roads, fileMode); - this.set().driver(driver); - - } - - public DownloadKit(Path goalPath, Integer roads, FileMode fileMode, SessionOptions driver) { - this(goalPath.toAbsolutePath().toString(), roads, fileMode, driver); - } - - public DownloadKit(String goalPath, Integer roads, FileMode fileMode, SessionOptions driver) { - init(goalPath, roads, fileMode); - this.set().driver(driver); - } - - private void init(String goalPath, Integer roads, FileMode fileMode) { - if (roads != null && roads > 0) this.roads = roads; - this.missions = new HashMap<>(); - this.waitingList = new LinkedBlockingQueue<>(); - this.threadMap = new ConcurrentHashMap<>(this.roads); - this.threads = new ThreadPoolExecutor(5, this.roads, 60, TimeUnit.SECONDS, new LinkedBlockingQueue<>()); - this.missionsNum = 0; - this.runningCount = 0; //正在运行的任务数 - this.stopPrinting = false; //用于控制显示线程停止 - this.goalPath = goalPath != null && !goalPath.trim().isEmpty() ? Utils.pathSetter(goalPath.trim()) : "."; - if (fileMode != null) this.fileMode = fileMode; - this.split = true; - this.blockSize = Utils.blockSizeSetter("50M"); - } - - /*** - * - * @return 用于设置打印和记录模式的对象 - */ - public Setter set() { - if (setter == null) setter = new Setter(this); - return setter; - } - - /** - * @return 可同时运行的线程数 - */ - public Integer roads() { - return this.roads; - } - - /** - * @return 返回连接失败时重试次数 - */ - public int retry() { - if (this.retry != null) return this.retry; - else if (this.page != null) return this.page.getRetryTimes(); - return 3; - } - - /** - * @return 返回连接失败时重试间隔 - */ - public double interval() { - if (this.interval != null) return this.interval; - else if (this.page != null) return this.page.getRetryInterval(); - return 5.0; - } - - /** - * @return 返回连接超时时间 - */ - public double timeout() { - if (this.timeout != null) return this.timeout; - else if (this.page != null) return this.page.timeout(); - return 20.0; - } - - /** - * @return 返回等待队列 - */ - public BlockingQueue waitingList() { - return this.waitingList; - } - - /** - * @return 返回用于保存默认连接设置的Session对象 - */ - public OkHttpClient session() { - return this.session; - } - - /** - * @return 返回是否有线程还在运行 - */ - public boolean isRunning() { - return this.runningCount > 0; - } - - /** - * @return map方式返回所有任务对象 - */ - public Map missions() { - return this.missions; - } - - /** - * @return 返回指定的编码格式 - */ - public String encoding() { - return this.encoding; - } - - - /** - * 添加一个下载任务并将其返回 - * - * @param fileUrl 文件网址 - * @return 任务对象 - */ - public Mission add(String fileUrl) { - return add(fileUrl, new HashMap<>()); - } - - /** - * 添加一个下载任务并将其返回 - * - * @param fileUrl 文件网址 - * @param params 连接参数 - * @return 任务对象 - */ - public Mission add(String fileUrl, Map params) { - return add(fileUrl, "", params); - } - - /** - * 添加一个下载任务并将其返回 - * - * @param fileUrl 文件网址 - * @param goalPath 保存路径 - * @return 任务对象 - */ - public Mission add(String fileUrl, String goalPath) { - return add(fileUrl, goalPath, new HashMap<>()); - } - - /** - * 添加一个下载任务并将其返回 - * - * @param fileUrl 文件网址 - * @param goalPath 保存路径 - * @param params 连接参数 - * @return 任务对象 - */ - public Mission add(String fileUrl, String goalPath, Map params) { - return add(fileUrl, goalPath == null || goalPath.isEmpty() ? null : goalPath, null, params); - } - - /** - * 添加一个下载任务并将其返回 - * - * @param fileUrl 文件网址 - * @param goalPath 保存路径 - * @param rename 重命名的文件名 - * @return 任务对象 - */ - public Mission add(String fileUrl, String goalPath, String rename) { - return add(fileUrl, goalPath, rename, new HashMap<>()); - } - - /** - * 添加一个下载任务并将其返回 - * - * @param fileUrl 文件网址 - * @param goalPath 保存路径 - * @param rename 重命名的文件名 - * @param params 连接参数 - * @return 任务对象 - */ - public Mission add(String fileUrl, String goalPath, String rename, Map params) { - return add(fileUrl, goalPath, rename, null, params); - } - - /** - * 添加一个下载任务并将其返回 - * - * @param fileUrl 文件网址 - * @param goalPath 保存路径 - * @param rename 重命名的文件名 - * @param suffix 重命名的文件后缀名 - * @return 任务对象 - */ - public Mission add(String fileUrl, String goalPath, String rename, String suffix) { - return add(fileUrl, goalPath, rename, suffix, new HashMap<>()); - } - - /** - * 添加一个下载任务并将其返回 - * - * @param fileUrl 文件网址 - * @param goalPath 保存路径 - * @param rename 重命名的文件名 - * @param suffix 重命名的文件后缀名 - * @param params 连接参数 - * @return 任务对象 - */ - public Mission add(String fileUrl, String goalPath, String rename, String suffix, Map params) { - return add(fileUrl, goalPath, rename, suffix, null, params); - } - - /** - * 添加一个下载任务并将其返回 - * - * @param fileUrl 文件网址 - * @param goalPath 保存路径 - * @param rename 重命名的文件名 - * @param suffix 重命名的文件后缀名 - * @return 任务对象 - */ - public Mission add(String fileUrl, String goalPath, String rename, String suffix, FileMode fileMode) { - return add(fileUrl, goalPath, rename, suffix, fileMode, null); - } - - /** - * 添加一个下载任务并将其返回 - * - * @param fileUrl 文件网址 - * @param goalPath 保存路径 - * @param rename 重命名的文件名 - * @param suffix 重命名的文件后缀名 - * @param fileMode 遇到同名文件时的处理方式,可选 'skip', 'overwrite', 'rename', 'add',默认跟随实例属性 - * @param params 连接参数 - * @return 任务对象 - */ - public Mission add(String fileUrl, String goalPath, String rename, String suffix, FileMode fileMode, Map params) { - return add(fileUrl, Paths.get(goalPath), rename, suffix, fileMode, null, params); - } - - /** - * 添加一个下载任务并将其返回 - * - * @param fileUrl 文件网址 - * @param goalPath 保存路径 - * @param rename 重命名的文件名 - * @param suffix 重命名的文件后缀名 - * @param fileMode 遇到同名文件时的处理方式,可选 'skip', 'overwrite', 'rename', 'add',默认跟随实例属性 - * @param split 是否允许多线程分块下载,为null则使用对象属性 - * @param params 连接参数 - * @return 任务对象 - */ - public Mission add(String fileUrl, String goalPath, String rename, String suffix, FileMode fileMode, Boolean split, Map params) { - return add(fileUrl, Paths.get(goalPath), rename, suffix, fileMode, split, params); - } - - /** - * 添加一个下载任务并将其返回 - * - * @param fileUrl 文件网址 - * @param goalPath 保存路径 - * @return 任务对象 - */ - public Mission add(String fileUrl, Path goalPath) { - return add(fileUrl, goalPath, new HashMap<>()); - } - - /** - * 添加一个下载任务并将其返回 - * - * @param fileUrl 文件网址 - * @param goalPath 保存路径 - * @param params 连接参数 - * @return 任务对象 - */ - public Mission add(String fileUrl, Path goalPath, Map params) { - return add(fileUrl, goalPath, null, params); - } - - /** - * 添加一个下载任务并将其返回 - * - * @param fileUrl 文件网址 - * @param goalPath 保存路径 - * @param rename 重命名的文件名 - * @return 任务对象 - */ - public Mission add(String fileUrl, Path goalPath, String rename) { - return add(fileUrl, goalPath, rename, new HashMap<>()); - } - - /** - * 添加一个下载任务并将其返回 - * - * @param fileUrl 文件网址 - * @param goalPath 保存路径 - * @param rename 重命名的文件名 - * @param params 连接参数 - * @return 任务对象 - */ - public Mission add(String fileUrl, Path goalPath, String rename, Map params) { - return add(fileUrl, goalPath, rename, null, params); - } - - /** - * 添加一个下载任务并将其返回 - * - * @param fileUrl 文件网址 - * @param goalPath 保存路径 - * @param rename 重命名的文件名 - * @param suffix 重命名的文件后缀名 - * @return 任务对象 - */ - public Mission add(String fileUrl, Path goalPath, String rename, String suffix) { - return add(fileUrl, goalPath, rename, suffix, new HashMap<>()); - } - - /** - * 添加一个下载任务并将其返回 - * - * @param fileUrl 文件网址 - * @param goalPath 保存路径 - * @param rename 重命名的文件名 - * @param suffix 重命名的文件后缀名 - * @param params 连接参数 - * @return 任务对象 - */ - public Mission add(String fileUrl, Path goalPath, String rename, String suffix, Map params) { - return add(fileUrl, goalPath, rename, suffix, null, params); - } - - /** - * 添加一个下载任务并将其返回 - * - * @param fileUrl 文件网址 - * @param goalPath 保存路径 - * @param rename 重命名的文件名 - * @param suffix 重命名的文件后缀名 - * @return 任务对象 - */ - public Mission add(String fileUrl, Path goalPath, String rename, String suffix, FileMode fileMode) { - return add(fileUrl, goalPath, rename, suffix, fileMode, null); - } - - /** - * 添加一个下载任务并将其返回 - * - * @param fileUrl 文件网址 - * @param goalPath 保存路径 - * @param rename 重命名的文件名 - * @param suffix 重命名的文件后缀名 - * @param fileMode 遇到同名文件时的处理方式,可选 'skip', 'overwrite', 'rename', 'add',默认跟随实例属性 - * @param params 连接参数 - * @return 任务对象 - */ - public Mission add(String fileUrl, Path goalPath, String rename, String suffix, FileMode fileMode, Map params) { - return add(fileUrl, goalPath, rename, suffix, fileMode, null, params); - } - - /** - * 添加一个下载任务并将其返回 - * - * @param fileUrl 文件网址 - * @param goalPath 保存路径 - * @param rename 重命名的文件名 - * @param suffix 重命名的文件后缀名 - * @param fileMode 遇到同名文件时的处理方式,可选 'skip', 'overwrite', 'rename', 'add',默认跟随实例属性 - * @param split 是否允许多线程分块下载,为null则使用对象属性 - * @param params 连接参数 - * @return 任务对象 - */ - public Mission add(String fileUrl, Path goalPath, String rename, String suffix, FileMode fileMode, Boolean split, Map params) { - this.missionsNum++; - this.runningCount++; - fileMode = fileMode == null ? this.fileMode : fileMode; - Mission mission = goalPath == null ? new Mission(String.valueOf(this.missionsNum), this, fileUrl, this.goalPath, rename, suffix, fileMode, split == null ? this.split : split, this.encoding, params) : new Mission(String.valueOf(this.missionsNum), this, fileUrl, goalPath, rename, suffix, fileMode, split == null ? this.split : split, this.encoding, params); - this.missions.put(this.missionsNum, mission); - this.runOrWait(mission); - return mission; - } - - - /** - * 以阻塞的方式下载一个文件并返回结果 - * - * @param fileUrl 文件网址 - * @return 任务结果和信息组成的数组 - */ - public String[] download(String fileUrl) { - return download(fileUrl, new HashMap<>()); - } - - /** - * 以阻塞的方式下载一个文件并返回结果 - * - * @param fileUrl 文件网址 - * @param params 连接参数 - * @return 任务结果和信息组成的数组 - */ - public String[] download(String fileUrl, Map params) { - return download(fileUrl, "", params); - } - - /** - * 以阻塞的方式下载一个文件并返回结果 - * - * @param fileUrl 文件网址 - * @param goalPath 保存路径 - * @return 任务结果和信息组成的数组 - */ - public String[] download(String fileUrl, String goalPath) { - return download(fileUrl, goalPath, new HashMap<>()); - } - - /** - * 以阻塞的方式下载一个文件并返回结果 - * - * @param fileUrl 文件网址 - * @param goalPath 保存路径 - * @param params 连接参数 - * @return 任务结果和信息组成的数组 - */ - public String[] download(String fileUrl, String goalPath, Map params) { - return download(fileUrl, goalPath == null || goalPath.isEmpty() ? null : goalPath, null, params); - } - - /** - * 以阻塞的方式下载一个文件并返回结果 - * - * @param fileUrl 文件网址 - * @param goalPath 保存路径 - * @param rename 重命名的文件名 - * @return 任务结果和信息组成的数组 - */ - public String[] download(String fileUrl, String goalPath, String rename) { - return download(fileUrl, goalPath, rename, new HashMap<>()); - } - - /** - * 以阻塞的方式下载一个文件并返回结果 - * - * @param fileUrl 文件网址 - * @param goalPath 保存路径 - * @param rename 重命名的文件名 - * @param params 连接参数 - * @return 任务结果和信息组成的数组 - */ - public String[] download(String fileUrl, String goalPath, String rename, Map params) { - return download(fileUrl, goalPath, rename, null, params); - } - - /** - * 以阻塞的方式下载一个文件并返回结果 - * - * @param fileUrl 文件网址 - * @param goalPath 保存路径 - * @param rename 重命名的文件名 - * @param suffix 重命名的文件后缀名 - * @return 任务结果和信息组成的数组 - */ - public String[] download(String fileUrl, String goalPath, String rename, String suffix) { - return download(fileUrl, goalPath, rename, suffix, new HashMap<>()); - } - - /** - * 以阻塞的方式下载一个文件并返回结果 - * - * @param fileUrl 文件网址 - * @param goalPath 保存路径 - * @param rename 重命名的文件名 - * @param suffix 重命名的文件后缀名 - * @param params 连接参数 - * @return 任务结果和信息组成的数组 - */ - public String[] download(String fileUrl, String goalPath, String rename, String suffix, Map params) { - return download(fileUrl, goalPath, rename, suffix, null, params); - } - - /** - * 以阻塞的方式下载一个文件并返回结果 - * - * @param fileUrl 文件网址 - * @param goalPath 保存路径 - * @param rename 重命名的文件名 - * @param suffix 重命名的文件后缀名 - * @return 任务结果和信息组成的数组 - */ - public String[] download(String fileUrl, String goalPath, String rename, String suffix, FileMode fileMode) { - return download(fileUrl, goalPath, rename, suffix, fileMode, null); - } - - /** - * 以阻塞的方式下载一个文件并返回结果 - * - * @param fileUrl 文件网址 - * @param goalPath 保存路径 - * @param rename 重命名的文件名 - * @param suffix 重命名的文件后缀名 - * @param fileMode 遇到同名文件时的处理方式,可选 'skip', 'overwrite', 'rename', 'add',默认跟随实例属性 - * @param params 连接参数 - * @return 任务结果和信息组成的数组 - */ - public String[] download(String fileUrl, String goalPath, String rename, String suffix, FileMode fileMode, Map params) { - return download(fileUrl, Paths.get(goalPath), rename, suffix, fileMode, true, params); - } - - /** - * 以阻塞的方式下载一个文件并返回结果 - * - * @param fileUrl 文件网址 - * @param goalPath 保存路径 - * @param rename 重命名的文件名 - * @param suffix 重命名的文件后缀名 - * @param fileMode 遇到同名文件时的处理方式,可选 'skip', 'overwrite', 'rename', 'add',默认跟随实例属性 - * @param showMsg 是否打印进度 - * @param params 连接参数 - * @return 任务结果和信息组成的数组 - */ - public String[] download(String fileUrl, String goalPath, String rename, String suffix, FileMode fileMode, boolean showMsg, Map params) { - return download(fileUrl, Paths.get(goalPath), rename, suffix, fileMode, split, params); - } - - /** - * 以阻塞的方式下载一个文件并返回结果 - * - * @param fileUrl 文件网址 - * @param goalPath 保存路径 - * @return 任务结果和信息组成的数组 - */ - public String[] download(String fileUrl, Path goalPath) { - return download(fileUrl, goalPath, new HashMap<>()); - } - - /** - * 以阻塞的方式下载一个文件并返回结果 - * - * @param fileUrl 文件网址 - * @param goalPath 保存路径 - * @param params 连接参数 - * @return 任务结果和信息组成的数组 - */ - public String[] download(String fileUrl, Path goalPath, Map params) { - return download(fileUrl, goalPath, null, params); - } - - /** - * 以阻塞的方式下载一个文件并返回结果 - * - * @param fileUrl 文件网址 - * @param goalPath 保存路径 - * @param rename 重命名的文件名 - * @return 任务结果和信息组成的数组 - */ - public String[] download(String fileUrl, Path goalPath, String rename) { - return download(fileUrl, goalPath, rename, new HashMap<>()); - } - - /** - * 以阻塞的方式下载一个文件并返回结果 - * - * @param fileUrl 文件网址 - * @param goalPath 保存路径 - * @param rename 重命名的文件名 - * @param params 连接参数 - * @return 任务结果和信息组成的数组 - */ - public String[] download(String fileUrl, Path goalPath, String rename, Map params) { - return download(fileUrl, goalPath, rename, null, params); - } - - /** - * 以阻塞的方式下载一个文件并返回结果 - * - * @param fileUrl 文件网址 - * @param goalPath 保存路径 - * @param rename 重命名的文件名 - * @param suffix 重命名的文件后缀名 - * @return 任务结果和信息组成的数组 - */ - public String[] download(String fileUrl, Path goalPath, String rename, String suffix) { - return download(fileUrl, goalPath, rename, suffix, new HashMap<>()); - } - - /** - * 以阻塞的方式下载一个文件并返回结果 - * - * @param fileUrl 文件网址 - * @param goalPath 保存路径 - * @param rename 重命名的文件名 - * @param suffix 重命名的文件后缀名 - * @param params 连接参数 - * @return 任务结果和信息组成的数组 - */ - public String[] download(String fileUrl, Path goalPath, String rename, String suffix, Map params) { - return download(fileUrl, goalPath, rename, suffix, null, params); - } - - /** - * 以阻塞的方式下载一个文件并返回结果 - * - * @param fileUrl 文件网址 - * @param goalPath 保存路径 - * @param rename 重命名的文件名 - * @param suffix 重命名的文件后缀名 - * @return 任务结果和信息组成的数组 - */ - public String[] download(String fileUrl, Path goalPath, String rename, String suffix, FileMode fileMode) { - return download(fileUrl, goalPath, rename, suffix, fileMode, null); - } - - /** - * 以阻塞的方式下载一个文件并返回结果 - * - * @param fileUrl 文件网址 - * @param goalPath 保存路径 - * @param rename 重命名的文件名 - * @param suffix 重命名的文件后缀名 - * @param fileMode 遇到同名文件时的处理方式,可选 'skip', 'overwrite', 'rename', 'add',默认跟随实例属性 - * @param params 连接参数 - * @return 任务结果和信息组成的数组 - */ - public String[] download(String fileUrl, Path goalPath, String rename, String suffix, FileMode fileMode, Map params) { - return download(fileUrl, goalPath, rename, suffix, fileMode, true, params); - } - - /** - * 以阻塞的方式下载一个文件并返回结果 - * - * @param fileUrl 文件网址 - * @param goalPath 保存路径 - * @param rename 重命名的文件名 - * @param suffix 重命名的文件后缀名 - * @param fileMode 遇到同名文件时的处理方式,可选 'skip', 'overwrite', 'rename', 'add',默认跟随实例属性 - * @param showMsg 是否打印进度 - * @param params 连接参数 - * @return 任务结果和信息组成的数组 - */ - public String[] download(String fileUrl, Path goalPath, String rename, String suffix, FileMode fileMode, boolean showMsg, Map params) { - String tmp = null; - if (showMsg) { - tmp = this.printMode; - this.printMode = null; - } - String[] wait = this.add(fileUrl, goalPath, rename, suffix, fileMode, false, params).wait(showMsg); - if (showMsg) this.printMode = tmp; - return wait; - } - - - /** - * 接收任务,有空线程则运行,没有则进入等待队列 - * - * @param mission 任务对象 - */ - private void runOrWait(BaseTask mission) { - Integer usableThread = this.getUsableThread(); - if (usableThread != null) { - Runnable runnable = () -> run(usableThread, mission); - Map map = this.threadMap.computeIfAbsent(usableThread, key -> new HashMap<>()); - map.put("thread", runnable); - map.put("mission", null); - threads.execute(runnable); - } else { - try { - this.waitingList.put(mission); - } catch (InterruptedException ignored) { - System.out.println("插入队列失败"); - } - } - } - - /** - * 线程函数 - * - * @param id 线程id - * @param mission 任务对象,Mission或Task - */ - - private void run(int id, BaseTask mission) { - - while (true) { - if (mission == null) if (this.waitingList.isEmpty()) break; - else try { - mission = waitingList.poll(500, TimeUnit.MILLISECONDS); - } catch (InterruptedException e) { - break; - } - threadMap.computeIfAbsent(id, key -> new HashMap<>()).put("mission", mission); - this.download(mission, id); - mission = null; - - } - threadMap.put(id, null); - } - - /** - * 根据id值获取一个任务 - * - * @param id 任务id - * @return 任务对象 - */ - public Mission getMission(int id) { - return this.missions.get(id); - } - - /** - * 根据id值获取一个任务 - * - * @param id 任务id - * @return 任务对象 - */ - public List getFailedMissions(int id) { - List list = new ArrayList<>(); - this.missions.forEach((a, b) -> { - if (b.getResult().equals("false")) list.add(b); - }); - return list; - } - - /** - * 等待所有或指定任务完成 - * - * @return 任务结果和信息组成的数组 - */ - public String[] waits() { - return wait(null); - } - - /** - * 等待所有或指定任务完成 - * - * @param mission 任务id,为null时等待所有任务结束 - * @return 任务结果和信息组成的数组 - */ - public String[] wait(Integer mission) { - return wait(mission, false); - } - - /** - * 等待所有或指定任务完成 - * - * @param mission 任务id,为null时等待所有任务结束 - * @param show 是否显示进度 - * @return 任务结果和信息组成的数组 - */ - public String[] wait(Integer mission, boolean show) { - return wait(mission, show, null); - } - - /** - * 等待所有或指定任务完成 - * - * @param mission 任务id,为null时等待所有任务结束 - * @param show 是否显示进度 - * @param timeout 超时时间,null或0为无限 - * @return 任务结果和信息组成的数组 - */ - - public String[] wait(Integer mission, boolean show, Double timeout) { - timeout = timeout == null ? 0 : timeout; - if (mission != null) return this.getMission(mission).wait(show, timeout); - else { - if (show) { - this.show(false); - } else { - long endTime = (long) (System.currentTimeMillis() + timeout * 1000); - while (this.isRunning() && (System.currentTimeMillis() < endTime || timeout == 0)) { - try { - Thread.sleep(100); - } catch (InterruptedException e) { - throw new RuntimeException(e); - } - } - } - return null; - } - } - - /** - * 取消所有等待中或执行中的任务 - */ - public void cancel() { - this.missions.values().forEach(Mission::cancel); - } - - /** - * 实时显示所有线程进度 - */ - public void show() { - show(true); - } - - /** - * 实时显示所有线程进度 - * - * @param asy 是否以异步方式显示 - */ - public void show(boolean asy) { - show(asy, false); - } - - /** - * 实时显示所有线程进度 - * - * @param asy 是否以异步方式显示 - * @param keep 任务列表为空时是否保持显示 - */ - public void show(boolean asy, boolean keep) { - if (asy) { - new Thread(() -> show(2.0, keep)).start(); - } else { - this.show(0.1, keep); - } - } - - private void show(double wait, boolean keep) { - this.stopPrinting = false; - if (keep) new Thread(this::stopShow).start(); - long endTime = (long) (System.currentTimeMillis() + wait); - while (!this.stopPrinting && (keep || isRunning() || System.currentTimeMillis() < endTime)) { - System.out.println("\33[K"); - System.out.println("等待任务数:" + waitingList.size()); - this.threadMap.forEach((k, v) -> { - BaseTask m = null; - if (v != null) if (v.get("mission") instanceof BaseTask) m = (BaseTask) v.get("mission"); - String path; - if (m != null) { - String[] items; - if (m instanceof Task) { - items = new String[]{String.valueOf(((Task) m).getMission().rate()), ((Task) m).mid()}; - } else { - items = new String[]{String.valueOf(((Mission) m).rate()), m.id()}; - } - path = "M" + items[1] + " " + items[0] + "% " + m; - } else { - path = "空闲"; - } - System.out.println("\033[K"); - System.out.println("线程" + k + ":" + path); - - - }); - System.out.println("\33[" + this.roads + 1 + "A\r"); - try { - Thread.sleep(400); - } catch (InterruptedException e) { - throw new RuntimeException(e); - } - } - System.out.println(); - } - - /** - * 生成response对象 - * - * @param url 目标url - * @param session 用于连接的Session对象 - * @param header 内置的headers参数 - * @param method 请求方式 - * @param encoding 编码格式 - * @param params 连接参数 - * @return 第一位为Response或None,第二位为出错信息或'Success' - */ - public ResponseConnect connect(String url, OkHttpClient session, CaseInsensitiveMap header, String method, String encoding, Map params) { - if (params.containsKey("headers")) { - Object o = params.get("headers"); - if (o instanceof Map) header.putAll((Map extends String, ?>) o); - params.put("headers", header); - } else { - params.put("headers", header); - } - Response response = null; - Exception e = null; - for (int i = 0; i < this.retry + 1; i++) { - try { - if ("get".equalsIgnoreCase(method)) { - Request.Builder builder = new Request.Builder(); - builder.get(); - Object headerObj = params.get("header"); - if (headerObj instanceof Map) for (Map.Entry, ?> entry : ((Map, ?>) headerObj).entrySet()) - builder.addHeader(entry.getKey().toString(), entry.getValue().toString()); - builder.setUrl$okhttp(HttpUrl.get(url)); - try (Response responses = session.newCall(builder.build()).execute()) { - response = responses; - } - } else if ("post".equalsIgnoreCase(method)) { - Request.Builder builder = new Request.Builder(); - builder.get(); - Object headerObj = params.get("header"); - if (headerObj instanceof Map) for (Map.Entry, ?> entry : ((Map, ?>) headerObj).entrySet()) - builder.addHeader(entry.getKey().toString(), entry.getValue().toString()); - builder.setUrl$okhttp(HttpUrl.get(url)); - Object o = params.get("json"); - if (o == null) o = params.get("body"); - if (o != null) - builder.setBody$okhttp(RequestBody.create(o.toString(), MediaType.get("application/json"))); - try (Response responses = session.newCall(builder.build()).execute()) { - response = responses; - } - } - if (response != null) { - return new ResponseConnect(Utils.setCharset(response, encoding), "Success"); - } - } catch (Exception es) { - e = es; - } - if (response != null && (response.code() == 403 || response.code() == 404)) { - break; - } - if (i < this.retry) { - try { - Thread.sleep((long) (this.interval * 1000)); - } catch (InterruptedException ex) { - throw new RuntimeException(ex); - } - } - } - - if (response == null) return new ResponseConnect(null, e == null ? "连接失败" : e.toString()); - if (response.code() != 200) return new ResponseConnect(response, "状态码:" + response.code()); - return new ResponseConnect(response, "返回成功"); - - } - - - /** - * @return 获取是否可用 - */ - private Integer getUsableThread() { - for (Map.Entry> entry : this.threadMap.entrySet()) - if (entry.getValue() != null) return entry.getKey(); - return null; - } - - - /** - * 设置停止打印的变量 - */ - private void stopShow() { - this.stopPrinting = false; - } - - /** - * 当任务完成时执行的操作 - * - * @param mission 完结的任务 - */ - public void _whenMissionDone(Mission mission) { - this.runningCount--; - if (Objects.equals(this.printMode, "all") || (Objects.equals(this.printMode, "failed") && Objects.equals(mission.getResult(), "false"))) - System.out.println("[" + Mission.RESULT_TEXTS.get(mission.getResult()) + "]" + mission.data().getUrl() + " " + mission.getInfo()); - if (Objects.equals(this.logMode, "all") || (Objects.equals(this.logMode, "failed") && Objects.equals(mission.getResult(), "false"))) { - Object[] data = {"下载结果", mission.data().getUrl(), mission.data().getRename(), mission.data().getParams()}; - this.logger.addData(data); - } - - } - - - /** - * 执行下载的方法,根据任务下载文件 - * - * @param missionOrTask 下载任务对象 - * @param threadId 线程号 - */ - private void download(BaseTask missionOrTask, int threadId) { - if (missionOrTask.isDone()) { - return; - } - - if (missionOrTask.getState().equals("cancel")) { - missionOrTask.setState("done"); - return; - } - - String fileUrl = missionOrTask.data().getUrl(); - - if (missionOrTask instanceof Task) { - Task task = (Task) missionOrTask; - Map params = new HashMap<>(missionOrTask.data().getParams()); - Object o = params.get("headers"); - if (o instanceof Map) { - ((Map) o).put("Range", "bytes=" + task.getRange().get(0) + "-" + task.getRange().get(1)); - } - ResponseConnect r = connect(fileUrl, task.getMission().getSession(), task.getMission().getHeaders(), task.getMission().getMethod(), task.getMission().getEncoding(), params); - - if (r.response != null) { - doDownload(r.response, task, false); - } else { - task._setDone("false", r.info); - } - return; - } - // ===================开始处理mission==================== - - Mission mission = (Mission) missionOrTask; - mission.setInfo("下载中"); - mission.setState("running"); - Map kwargs = missionOrTask.data().getParams(); - if (Objects.equals(printMode, "all")) { - System.out.println("开始下载:" + mission.data().getUrl()); - } - if (Objects.equals(logMode, "all")) { - logger.addData("开始下载", mission.data().getUrl()); - } - - String rename = mission.data().getRename(); - String suffix = mission.data().getSuffix(); - String goalPath = mission.data().getGoalPath(); - FileMode fileExists = mission.data().getFileExists(); - boolean split = mission.data().isSplit(); - - Path goalPathObj = Paths.get(goalPath); - //按windows规则去除路径中的非法字符 - goalPath = goalPathObj.getRoot() + goalPathObj.subpath(1, goalPathObj.getNameCount()).toString().replaceAll("[*:|<>?\"]", "").trim(); - - goalPathObj = Paths.get(goalPath).toAbsolutePath(); - goalPathObj.toFile().mkdirs(); - goalPath = goalPathObj.toString(); - - if (fileExists.equals(FileMode.SKIP) && rename != null && !rename.isEmpty()) { - Path tmp = goalPathObj.resolve(rename); - if (Files.exists(tmp) && Files.isRegularFile(tmp)) { - mission.setFileName(rename); - mission._setPath(tmp.toString()); - mission._setDone("skipped", mission.path()); - return; - } - } - - ResponseConnect r = connect(fileUrl, mission.getSession(), mission.getHeaders(), mission.getMethod(), mission.getEncoding(), kwargs); - - if (mission.isDone()) { - return; - } - - if (r.response == null) { - mission._breakMission("false", r.info); - return; - } - //-------------------获取文件信息------------------- - Map fileInfo = Utils.getFileInfo(r.response, goalPath, rename, suffix, fileExists, mission.getEncoding(), this.lock); - long fileSize = Long.parseLong(fileInfo.get("size").toString()); - Path fullPath = Paths.get((String) fileInfo.get("path")); - mission._setPath(fullPath); - mission.setFileName(fullPath.getFileName().toString()); - mission.setSize(fileSize); - - if ((boolean) fileInfo.get("skip")) { - mission._setDone("skipped", mission.path()); - return; - } - - fullPath = Paths.get((String) fileInfo.get("path")); - if (fileExists.equals("add") && Files.exists(fullPath)) { - try { - mission.data().setOffset(Files.size(fullPath)); - } catch (IOException e) { - throw new RuntimeException(e); - } - } - - boolean first = false; - if (split && fileSize > this.blockSize && Objects.equals(r.response.headers().get("Accept-Ranges"), "bytes")) { - first = true; - List> chunks = new ArrayList<>(); - for (long s = 0; s < fileSize; s += blockSize) { - double e = Math.min(s + blockSize, fileSize) - 1; - List objects = new ArrayList<>(); - objects.add(s); - objects.add(e); - chunks.add(objects); - } - chunks.get(chunks.size() - 1).set(1, -1); - Task task1 = new Task(mission, chunks.get(0), "1/" + chunks.size(), new BigDecimal(chunks.get(0).get(1).toString()).subtract(new BigDecimal(chunks.get(0).get(0).toString())).longValue()); - mission.setTasksCount(chunks.size()); - mission.setTasks(Collections.singletonList(task1)); - - for (int ind = 2; ind <= chunks.size(); ind++) { - List chunk = chunks.get(ind - 1); - long s = fileSize - Long.parseLong(chunk.get(0).toString()); - Task task = new Task(mission, chunk, ind + "/" + chunks.size(), s); - mission.getTasks().add(task); - runOrWait(task); - } - } else { - Task task1 = new Task(mission, null, "1/1", fileSize); - mission.getTasks().add(task1); - } - - threadMap.get(threadId).put("mission", mission.getTasks().get(0)); - doDownload(r.response, mission.getTasks().get(0), first); - } - - - @AllArgsConstructor - public static class ResponseConnect { - private Response response; - private String info; - } - - - /** - * 执行下载任务 - * - * @param response 响应对象 - * @param task 任务对象 - * @param first 是否第一个分块 - */ - public static void doDownload(Response response, Task task, boolean first) { - if (task.isDone() || task.getMission().isDone()) { - return; - } - - task.setStates(null, "下载中", "running"); - int blockSize = 131072; // 128k - String result = null; - String info = null; - - try (ResponseBody responseBody = response.body()) { - if (first) { // 第一个分块 - if (Long.parseLong(task.getRange().get(1).toString()) <= blockSize || Long.parseLong(task.getRange().get(1).toString()) % blockSize != 0) { - byte[] content = new byte[0]; - if (responseBody != null) { - content = responseBody.bytes(); - } - task.addData(content, task.getMission().data().getOffset()); - if ("cancel".equals(task.getState()) || "done".equals(task.getState())) { - result = "canceled"; - task.clearCache(); - } - } else { - long blocks = Long.parseLong(task.getRange().get(1).toString()) / blockSize; - for (int b = 0; b < blocks; b++) { - byte[] content = new byte[0]; - if (responseBody != null) { - content = responseBody.bytes(); - } - task.addData(content, (long) b * blockSize + task.getMission().data().getOffset()); - if ("cancel".equals(task.getState()) || "done".equals(task.getState())) { - result = "canceled"; - task.clearCache(); - break; - } - } - - if ("cancel".equals(task.getState()) || "done".equals(task.getState())) { - result = "canceled"; - task.clearCache(); - } else { - byte[] content = new byte[0]; - if (responseBody != null) { - content = responseBody.bytes(); - } - task.addData(content, blocks * blockSize + task.getMission().data().getOffset()); - } - } - } else { - if (task.getRange() == null) { // 不分块 - byte[] chunk = new byte[0]; - if (responseBody != null) { - chunk = responseBody.bytes(); - } - for (byte b : chunk) { - if ("cancel".equals(task.getState()) || "done".equals(task.getState())) { - result = "canceled"; - task.clearCache(); - break; - } - task.addData(new byte[]{b}, null); - } - } else if (task.getRange().get(1) == null) { // 结尾的数据块 - long begin = Long.parseLong(task.getRange().get(0).toString()); - if (responseBody != null) { - for (byte b : responseBody.bytes()) { - if ("cancel".equals(task.getState()) || "done".equals(task.getState())) { - result = "canceled"; - task.clearCache(); - break; - } - task.addData(new byte[]{b}, begin + task.getMission().data().getOffset()); - begin++; - } - } - } else { // 有始末数字的数据块 - long begin = (long) task.getRange().get(0); - long end = (long) task.getRange().get(1); - int num = (int) ((end - begin) / blockSize); - int ind = 1; - if (responseBody != null) { - for (byte b : responseBody.bytes()) { - if ("cancel".equals(task.getState()) || "done".equals(task.getState())) { - result = "canceled"; - task.clearCache(); - break; - } - task.addData(new byte[]{b}, begin + task.getMission().data().getOffset()); - if (ind <= num) { - begin += blockSize; - } - ind++; - } - } - } - } - } catch (IOException e) { - result = "failed"; - info = "下载失败。" + response.code() + " " + e.getMessage(); - } finally { - response.close(); - } - - if (result == null) { - result = "success"; - info = String.valueOf(task.path()); - } - - task._setDone(result, info); - } -} diff --git a/java/src/main/java/com/ll/DownloadKit/FileExists.java b/java/src/main/java/com/ll/DownloadKit/FileExists.java deleted file mode 100644 index f3f684f..0000000 --- a/java/src/main/java/com/ll/DownloadKit/FileExists.java +++ /dev/null @@ -1,46 +0,0 @@ -package com.ll.DownloadKit; - -import lombok.AllArgsConstructor; - -/** - * 用于设置存在同名文件时处理方法 - * - * @author 陆 - * @address click - */ -@AllArgsConstructor -public class FileExists { - private final Setter setter; - - public void set(FileMode fileMode) { - this.setter.downloadKit.fileMode = fileMode; - } - - /** - * 设为跳过 - */ - public void skip() { - this.setter.downloadKit.fileMode = FileMode.SKIP; - } - - /** - * 设为重命名,文件名后加序号 - */ - public void rename() { - this.setter.downloadKit.fileMode = FileMode.rename; - } - - /** - * 设为覆盖 - */ - public void overwrite() { - this.setter.downloadKit.fileMode = FileMode.overwrite; - } - - /** - * 设为追加 - */ - public void add() { - this.setter.downloadKit.fileMode = FileMode.add; - } -} diff --git a/java/src/main/java/com/ll/DownloadKit/FileMode.java b/java/src/main/java/com/ll/DownloadKit/FileMode.java deleted file mode 100644 index a9e2830..0000000 --- a/java/src/main/java/com/ll/DownloadKit/FileMode.java +++ /dev/null @@ -1,18 +0,0 @@ -package com.ll.DownloadKit; - -import lombok.Getter; - -/** - * @author 陆 - * @address click - */ -@Getter -public enum FileMode { - ADD("add"), add(ADD.value), SKIP("skip"), skip(SKIP.value), RENAME("rename"), rename(RENAME.value), OVERWRITE("overwrite"), overwrite(OVERWRITE.value), a(ADD.value), A(ADD.value), S(SKIP.value), s(SKIP.value), r(RENAME.value), R(RENAME.value), o(OVERWRITE.value), O(OVERWRITE.value); - private final String value; - - FileMode(String value) { - this.value = value; - } - -} diff --git a/java/src/main/java/com/ll/DownloadKit/LogSet.java b/java/src/main/java/com/ll/DownloadKit/LogSet.java deleted file mode 100644 index a077f4d..0000000 --- a/java/src/main/java/com/ll/DownloadKit/LogSet.java +++ /dev/null @@ -1,86 +0,0 @@ -package com.ll.DownloadKit; - -import com.ll.DataRecorder.Recorder; -import lombok.AllArgsConstructor; - -import java.nio.file.Path; - -/** - * 用于设置信息打印和记录日志方式 - * - * @author 陆 - * @address click - */ -@AllArgsConstructor -public class LogSet { - private final Setter setter; - - /** - * 设置日志文件路径 - * - * @param path 文件路径,可以是str或Path - */ - public void path(String path) { - if (this.setter.downloadKit.getLogger() != null) this.setter.downloadKit.getLogger().record(); - this.setter.downloadKit.logger= new Recorder(path); - } - - /** - * 设置日志文件路径 - * - * @param path 文件路径,可以是str或Path - */ - - public void path(Path path) { - if (this.setter.downloadKit.getLogger() != null) this.setter.downloadKit.getLogger().record(); - this.setter.downloadKit.logger= new Recorder(path); - } - - /** - * 打印所有信息 - */ - public void printAll() { - this.setter.downloadKit.printMode= "all"; - } - - /** - * 只有在下载失败时打印信息 - */ - public void printFailed() { - this.setter.downloadKit.printMode= "failed"; - } - - - /** - * 不打印任何信息 - */ - public void printNull() { - this.setter.downloadKit.printMode= null; - } - - /** - * 记录所有信息 - */ - public void logAll() { - if (this.setter.downloadKit.getLogger() == null) throw new RuntimeException("请先用logPath()设置log文件路径。"); - this.setter.downloadKit.logMode= "all"; - } - - /** - * 只记录下载失败的信息 - */ - public void logFailed() { - if (this.setter.downloadKit.getLogger() == null) throw new RuntimeException("请先用logPath()设置log文件路径。"); - this.setter.downloadKit.logMode= "failed"; - } - - /** - * 不进行记录 - */ - public void logNull() { - if (this.setter.downloadKit.getLogger() == null) throw new RuntimeException("请先用logPath()设置log文件路径。"); - this.setter.downloadKit.logMode= null; - } - - -} diff --git a/java/src/main/java/com/ll/DownloadKit/Setter.java b/java/src/main/java/com/ll/DownloadKit/Setter.java deleted file mode 100644 index d904559..0000000 --- a/java/src/main/java/com/ll/DownloadKit/Setter.java +++ /dev/null @@ -1,197 +0,0 @@ -package com.ll.DownloadKit; - -import com.ll.DrissonPage.base.BasePage; -import com.ll.DrissonPage.config.SessionOptions; -import com.ll.DrissonPage.page.SessionPage; -import com.ll.DrissonPage.page.WebPage; -import com.ll.DrissonPage.units.HttpClient; -import lombok.AllArgsConstructor; -import okhttp3.Interceptor; -import okhttp3.OkHttpClient; -import okhttp3.Request; -import okhttp3.Response; -import org.apache.http.Header; -import org.jetbrains.annotations.NotNull; - -import java.io.IOException; -import java.net.Proxy; -import java.nio.charset.Charset; -import java.nio.file.Path; -import java.util.Collection; -import java.util.HashMap; - -/** - * @author 陆 - * @address click - */ -@AllArgsConstructor -public class Setter { - protected final DownloadKit downloadKit; - - - /** - * - - * 设置Session对象 - * - * @param driver Session对象或DrissionPage的页面对象 - */ - - public void driver(Object driver) { - if (driver == null) { - this.downloadKit.session = new OkHttpClient(); - return; - } - if (driver instanceof OkHttpClient) { - this.downloadKit.session = (OkHttpClient) driver; - return; - } - if (driver instanceof SessionOptions) { - HttpClient httpClient = ((SessionOptions) driver).makeSession(); - Collection extends Header> headers = httpClient.getHeaders(); - this.downloadKit.session = this.downloadKit.session().newBuilder().addInterceptor(new Interceptor() { - @NotNull - @Override - public Response intercept(@NotNull Interceptor.Chain chain) throws IOException { - Request request = chain.request(); - Request.Builder builder1 = request.newBuilder(); - if (!headers.isEmpty()) headers.forEach((a) -> builder1.addHeader(a.getName(), a.getValue())); - return chain.proceed(request); - } - }).build(); - } else if (driver instanceof BasePage) { - if (driver instanceof SessionPage) this.downloadKit.session = ((SessionPage) driver).session(); - else if (driver instanceof WebPage) this.downloadKit.session = ((WebPage) driver).session(); - else this.downloadKit.session = new OkHttpClient(); - this.downloadKit.page = ((BasePage>) driver); - } else { - throw new IllegalArgumentException("类型只能为OkHttpClient SessionOptions BasePage"); - } - } - - /** - * 设置可同时运行的线程数 - * - * @param num 线程数量 - */ - public void roads(int num) { - if (this.downloadKit.isRunning()) { - System.out.println("有任务未完成时不能改变roads。"); - return; - } - if (num != this.downloadKit.roads()) { - this.downloadKit.roads = num; - this.downloadKit.getThreads().setMaximumPoolSize(num); - this.downloadKit.threadMap = new HashMap<>(num); - } - } - - /** - * 设置连接失败时重试次数 - * - * @param times 重试次数 - */ - public void retry(int times) { - if (times < 0) throw new IllegalArgumentException("times参数过于小"); - this.downloadKit.retry = times; - } - - /** - * 设置连接失败时重试间隔 - * - * @param seconds 连接失败时重试间隔(秒) - */ - - public void interval(double seconds) { - if (seconds < 0) throw new IllegalArgumentException("seconds参数过于小"); - this.downloadKit.interval = seconds; - } - - /** - * 设置连接超时时间 - * - * @param timeout 超时时间(秒) - */ - public void timeout(double timeout) { - if (timeout < 0) throw new IllegalArgumentException("timeout参数过于小"); - this.downloadKit.timeout = timeout; - } - - /** - * 设置文件保存路径 - * - * @param path 文件路径,可以是str或Path - */ - public void goalPath(Path path) { - this.downloadKit.goalPath = path.toAbsolutePath().toString(); - } - - /** - * 设置文件保存路径 - * - * @param path 文件路径,可以是str或Path - */ - public void goalPath(String path) { - this.downloadKit.goalPath = path; - } - - /** - * 设置大文件是否分块下载 - * - * @param onOff 代表开关 - */ - public void split(boolean onOff) { - this.downloadKit.split = onOff; - } - - /** - * 设置分块大小 - * - * @param size 单位为字节,可用'K'、'M'、'G'为单位,如'50M' - */ - public void blockSize(String size) { - this.downloadKit.blockSize = Utils.blockSizeSetter(size); - } - - /** - * 设置分块大小 - * - * @param size 单位为字节,可用'K'、'M'、'G'为单位,如'50M' - */ - public void blockSize(int size) { - this.downloadKit.blockSize = Utils.blockSizeSetter(size); - } - - /** - * 设置代理地址及端口,例:'127.0.0.1:1080' 创建方式 new Proxy(Proxy.Type.HTTP,new InetSocketAddress("127.0.0.1",80)) - */ - public void proxy(Proxy proxy) { - OkHttpClient.Builder builder = this.downloadKit.session.newBuilder(); - builder.setProxy$okhttp(proxy); - this.downloadKit.session = builder.build(); - } - - /** - * 设置编码 - * - * @param encoding 编码名称 - */ - public void encoding(Charset encoding) { - this.downloadKit.encoding = encoding.name(); - } - - /** - * 设置编码 - * - * @param encoding 编码名称 使用Charset.forName去校验 - */ - public void encoding(String encoding) { - this.downloadKit.encoding = Charset.forName(encoding).name(); - } - - /** - * 设置编码 为空 - */ - public void encoding() { - this.downloadKit.encoding = null; - } -} diff --git a/java/src/main/java/com/ll/DownloadKit/Utils.java b/java/src/main/java/com/ll/DownloadKit/Utils.java deleted file mode 100644 index 0e12f7b..0000000 --- a/java/src/main/java/com/ll/DownloadKit/Utils.java +++ /dev/null @@ -1,329 +0,0 @@ -package com.ll.DownloadKit; - -import com.ll.DataRecorder.Tools; -import okhttp3.*; -import org.jetbrains.annotations.NotNull; - -import java.io.IOException; -import java.io.UnsupportedEncodingException; -import java.net.URLDecoder; -import java.nio.charset.Charset; -import java.nio.file.Files; -import java.nio.file.Path; -import java.nio.file.Paths; -import java.util.*; -import java.util.concurrent.ThreadLocalRandom; -import java.util.concurrent.locks.Lock; -import java.util.regex.Matcher; -import java.util.regex.Pattern; - -/** - * @author 陆 - * @address click - */ -public class Utils { - private static final Map blockSizeMap; - private static final Map FILE_EXISTS_MODE; - - static { - blockSizeMap = new HashMap<>(); - blockSizeMap.put("b", 1L); - blockSizeMap.put("k", 1024L); - blockSizeMap.put("m", 1048576L); - blockSizeMap.put("g", 21_474_836_480L); - - FILE_EXISTS_MODE = new HashMap<>(); - FILE_EXISTS_MODE.put("rename", "rename"); - FILE_EXISTS_MODE.put("overwrite", "overwrite"); - FILE_EXISTS_MODE.put("skip", "skip"); - FILE_EXISTS_MODE.put("add", "add"); - FILE_EXISTS_MODE.put("r", "rename"); - FILE_EXISTS_MODE.put("o", "overwrite"); - FILE_EXISTS_MODE.put("s", "skip"); - FILE_EXISTS_MODE.put("a", "add"); - } - - public static long blockSizeSetter(Object val) { - if (val instanceof Integer || val instanceof Long) { - if (Long.parseLong(val.toString()) > 0) { - return (int) val; - } else { - throw new IllegalArgumentException(val + "数字需要大于0"); - } - } - if (val instanceof String && val.toString().length() >= 2) { - String string = val.toString(); - long num = Long.parseLong(string.substring(0, string.length() - 2)); - Long unit = blockSizeMap.get(String.valueOf(string.charAt(string.length() - 1)).toLowerCase()); - if (unit != null && num > 0) { - return unit * num; - } else { - throw new IllegalArgumentException("单位只支持B、K、M、G,数字必须为大于0的整数。"); - - } - } else { - throw new IllegalArgumentException("只能传入int或str,数字必须为大于0的整数。"); - } - } - - public static String pathSetter(Object path) { - if (path instanceof String) return (String) path; - if (path instanceof Path) return ((Path) path).toAbsolutePath().toString(); - throw new IllegalArgumentException("只能传入Path或str"); - - } - - - - /** - * 设置Response对象的编码 - * - * @param response Response对象 - * @param encoding 指定的编码格式 - * @return 设置编码后的Response对象 - */ - public static Response setCharset(Response response, String encoding) { - if (encoding != null && !encoding.isEmpty()) { - response = setEncoding(response, encoding); - return response; - } - - // 在headers中获取编码 - String contentType = response.headers("content-type").toString().toLowerCase(); - if (!contentType.endsWith(";")) { - contentType += ";"; - } - - String charset = findCharset(contentType); - if (charset != null) { - response = setEncoding(response, charset); - return response; - } - - // 在headers中获取不到编码,且如果是网页 - if (contentType.replace(" ", "").startsWith("text/html")) { - Matcher matcher = null; - try { - if (response.body() != null) { - matcher = Pattern.compile("]+).*?>").matcher(response.body().string()); - } - } catch (IOException e) { - throw new RuntimeException(e); - } - - if (matcher != null && matcher.find()) { - charset = matcher.group(1); - } - - response = setEncoding(response, charset); - } - - return response; - } - - private static Response setEncoding(Response response, String charset) { - if (charset != null && !charset.isEmpty()) { - Response.Builder build = response.newBuilder(); - ResponseBody body = response.body(); - if (body != null) if (body.contentType() != null) - Objects.requireNonNull(body.contentType()).charset(Charset.forName(charset)); - return build.build(); - } - return response; - } - - private static String findCharset(String contentType) { - Matcher matcher = Pattern.compile("charset[=: ]*(.*)?;?").matcher(contentType); - if (matcher.find()) { - return matcher.group(1); - } - return null; - } - - /** - * 获取文件信息,大小单位为byte - * 包括:size、path、skip - * - * @param response Response对象 - * @param goalPath 目标文件夹 - * @param rename 重命名 - * @param suffix 重命名后缀名 - * @param fileExists 存在重名文件时的处理方式 - * @param encoding 编码格式 - * @param lock 线程锁 - * @return 文件名、文件大小、保存路径、是否跳过 - */ - public static Map getFileInfo(Response response, String goalPath, String rename, String suffix, FileMode fileExists, String encoding, Lock lock) { - // ------------获取文件大小------------ - long fileSize = Optional.ofNullable(response.headers().get("Content-Length")).map(Long::parseLong).orElse(-1L); - - // ------------获取网络文件名------------ - String fileName = getFileName(response, encoding); - - // ------------获取保存路径------------ - Path goalPathObj = Paths.get(goalPath); - goalPath = goalPathObj.getRoot() + goalPathObj.subpath(1, goalPathObj.getNameCount()).toString().replaceAll("[*:|<>?\"]", "").trim(); - - Path goalPathAbsolute = Paths.get(goalPath).toAbsolutePath(); - goalPathAbsolute.toFile().mkdirs(); - goalPath = goalPathAbsolute.toString(); - - // ------------获取保存文件名------------ - // -------------------重命名------------------- - String fullFileName; - if (rename != null) { - if (suffix != null) { - fullFileName = suffix.isEmpty() ? rename : rename + "." + suffix; - } else { - String[] tmp = fileName.split("\\.", 2); - String extName = tmp.length > 1 ? "." + tmp[1] : ""; - tmp = rename.split("\\.", 2); - String extRename = tmp.length > 1 ? "." + tmp[1] : ""; - fullFileName = extRename.equals(extName) ? rename : rename + extName; - } - } else if (suffix != null) { - String[] tmp = fileName.split("\\.", 2); - fullFileName = suffix.isEmpty() ? tmp[0] : tmp[0] + "." + suffix; - } else { - fullFileName = fileName; - } - - fullFileName = Tools.makeValidName(fullFileName); - - // -------------------生成路径------------------- - boolean skip = false; - boolean create = true; - Path fullPath = Paths.get(goalPath, fullFileName); - - lock.lock(); - try { - if (Files.exists(fullPath)) { - if (FileMode.RENAME.equals(fileExists)) { - fullPath = Tools.getUsablePath(fullPath.toString()); - } else if (FileMode.SKIP.equals(fileExists)) { - skip = true; - create = false; - } else if (FileMode.OVERWRITE.equals(fileExists)) { - try { - Files.delete(fullPath); - } catch (IOException e) { - e.printStackTrace(); - } - } else if (FileMode.ADD.equals(fileExists)) { - create = false; - } - } - - if (create) { - try { - Files.createFile(fullPath); - } catch (IOException e) { - e.printStackTrace(); - } - } - } finally { - lock.unlock(); - } - - Map fileInfo = new HashMap<>(); - fileInfo.put("size", fileSize); - fileInfo.put("path", fullPath); - fileInfo.put("skip", skip); - - return fileInfo; - } - - - /** - * 从headers或url中获取文件名,如果获取不到,生成一个随机文件名 - * - * @param response 返回的response - * @param encoding 在headers获取时指定编码格式 - * @return 下载文件的文件名 - */ - private static String getFileName(Response response, String encoding) { - String fileName = ""; - String charset = "utf-8"; - - String contentDisposition = Objects.requireNonNull(response.header("content-disposition", "")).replace(" ", ""); - - if (!contentDisposition.isEmpty()) { - String txt = matchFilename(contentDisposition); - if (txt != null) { - if (txt.contains("''")) { - String[] parts = txt.split("''", 2); - charset = parts[0]; - fileName = parts[1]; - } else { - fileName = txt; - } - } else { - txt = matchFilename(contentDisposition, "filename"); - if (txt != null) { - fileName = txt; - if (response.body() != null) { - charset = encoding != null ? encoding : Objects.requireNonNull(Objects.requireNonNull(response.body().contentType()).charset()).toString(); - } - } - } - - fileName = fileName.replace("'", ""); - } - - if (fileName.isEmpty()) { - Paths.get(response.request().url().encodedPath()); - fileName = Paths.get(response.request().url().encodedPath()).getFileName().toString().split("\\?")[0]; - } - - if (fileName.isEmpty()) { - fileName = "untitled_" + System.currentTimeMillis() + "_" + ThreadLocalRandom.current().nextInt(100); - } - - charset = charset.isEmpty() ? "utf-8" : charset; - try { - return URLDecoder.decode(fileName, charset); - } catch (UnsupportedEncodingException e) { - return null; - } - } - - private static String matchFilename(String contentDisposition) { - Matcher matcher = Pattern.compile("filename\\*?=\"?([^\";]+)\"?").matcher(contentDisposition); - if (matcher.find()) { - String[] parts = matcher.group(1).split("''", 2); - return parts.length == 2 ? parts[0] + parts[1] : parts[0]; - } - return null; - } - - private static String matchFilename(String contentDisposition, String pattern) { - Matcher matcher = Pattern.compile(pattern + "=\"?([^\";]+)\"?").matcher(contentDisposition); - if (matcher.find()) { - return matcher.group(1); - } - return null; - } - - /** - * 设置OkHttpClient.Builder对象的cookies - * - * @param builder OkHttpClient.Builder对象 - * @param cookies cookies信息 - */ - public static void setSessionCookies(OkHttpClient.Builder builder, List cookies) { - for (Cookie cookie : cookies) { - builder.setCookieJar$okhttp(new CookieJar() { - @Override - public void saveFromResponse(@NotNull HttpUrl httpUrl, @NotNull List list) { - list.add(cookie); - } - - @NotNull - @Override - public List loadForRequest(@NotNull HttpUrl httpUrl) { - return new ArrayList<>(); - } - }); - } - } -} diff --git a/java/src/main/java/com/ll/DownloadKit/mission/BaseTask.java b/java/src/main/java/com/ll/DownloadKit/mission/BaseTask.java deleted file mode 100644 index 4be6dee..0000000 --- a/java/src/main/java/com/ll/DownloadKit/mission/BaseTask.java +++ /dev/null @@ -1,131 +0,0 @@ -package com.ll.DownloadKit.mission; - -import lombok.Getter; -import lombok.Setter; - -import java.util.Map; - -/** - * 任务类基类 - * - * @author 陆 - * @address click - */ -public class BaseTask { - protected final static String DONE = "done"; - public final static Map RESULT_TEXTS = Map.of("success", "成功", "skipped", "跳过", "canceled", "取消", "false", "失败", "null", "未知"); - /** - * 任务id - */ - private final String id; - @Getter - @Setter - - protected String state = "waiting"; // 'waiting'、'running'、'done' - @Getter - protected String result = "null"; //'success'、'skipped'、'canceled'、'false'、'null' - @Getter - @Setter - protected String info = "等待下载"; // 信息 - - /** - * @param id 任务id - */ - public BaseTask(String id) { - this.id = id; - } - - /** - * @return 返回任务或子任务id - */ - public String id() { - return this.id; - } - - /** - * @return 返回任务数据 - */ - - public MissionData data() { - return null; - } - - /** - * @return 返回任务是否结束 - */ - public boolean isDone() { - return "done".equalsIgnoreCase(this.state) || "cancel".equalsIgnoreCase(this.state); - } - - /** - * 设置任务结果值 - */ - public void setStates() { - setStates(result, info, "done"); - } - - /** - * 设置任务结果值 - * - * @param result 结果:'success'、'skipped'、'canceled'、false、null - */ - public void setStates(String result) { - setStates(result, null); - } - - /** - * 设置任务结果值 - * - * @param result 结果:'success'、'skipped'、'canceled'、false、null - * @param info 任务信息 - */ - public void setStates(String result, String info) { - setStates(result, info, "done"); - } - - /** - * 设置任务结果值 - * - * @param result 结果:'success'、'skipped'、'canceled'、false、null - * @param info 任务信息 - * @param state 任务状态:'waiting'、'running'、'done' - */ - public void setStates(String result, String info, String state) { - if (result == null) result = "null"; - result = result.toLowerCase().trim(); - switch (result) { - case "success": - this.result = "success"; - break; - case "skipped": - this.result = "skipped"; - break; - case "canceled": - this.result = "canceled"; - break; - case "false": - this.result = "False"; - break; - case "null": - this.result = "null"; - break; - default: - this.result = "null"; - break; - } - this.info = info; - if (state == null) state = "done"; - state = state.toLowerCase().trim(); - switch (state) { - case "running": - this.state = "running"; - break; - case "waiting": - this.state = "waiting"; - break; - default: - this.state = "done"; - break; - } - } -} diff --git a/java/src/main/java/com/ll/DownloadKit/mission/Mission.java b/java/src/main/java/com/ll/DownloadKit/mission/Mission.java deleted file mode 100644 index 353f09b..0000000 --- a/java/src/main/java/com/ll/DownloadKit/mission/Mission.java +++ /dev/null @@ -1,401 +0,0 @@ -package com.ll.DownloadKit.mission; - -import com.alibaba.fastjson.JSON; -import com.ll.DataRecorder.ByteRecorder; -import com.ll.DownloadKit.DownloadKit; -import com.ll.DownloadKit.FileMode; -import com.ll.DownloadKit.Utils; -import kotlin.Pair; -import lombok.Getter; -import lombok.Setter; -import okhttp3.*; -import org.apache.commons.collections4.map.CaseInsensitiveMap; -import org.jetbrains.annotations.NotNull; - -import java.io.IOException; -import java.lang.reflect.Field; -import java.math.BigDecimal; -import java.math.RoundingMode; -import java.net.URI; -import java.nio.file.Files; -import java.nio.file.Path; -import java.nio.file.Paths; -import java.util.ArrayList; -import java.util.List; -import java.util.Map; -import java.util.Objects; - -/** - * 任务类 - * - * @author 陆 - * @address click - */ -public class Mission extends BaseTask { - - @Setter - protected String fileName; - private final MissionData data; - private String path; - private ByteRecorder recorder; - @Setter - private Long size; - private int doneTasksCount = 0; - @Setter - private int tasksCount = 1; - @Setter - @Getter - private List tasks; - /** - * 所属DownloadKit对象 - */ - private final DownloadKit downloadKit; - @Getter - private OkHttpClient session; - @Getter - private CaseInsensitiveMap headers; - @Getter - private String method; - @Getter - - private String encoding; - - /** - * @param id 任务id - * @param downloadKit 所属DownloadKit对象 - * @param fileUrl 文件网址 - * @param goalPath 保存文件夹路径 - * @param rename 重命名 - * @param suffix 重命名后缀名 - * @param fileExists 存在同名文件处理方式 - * @param split 是否分块下载 - * @param encoding 编码格式 - * @param params 连接参数 - */ - public Mission(String id, DownloadKit downloadKit, String fileUrl, String goalPath, String rename, String suffix, FileMode fileExists, boolean split, String encoding, Map params) { - super(id); - this.downloadKit = downloadKit; - this.size = null; - this.tasks = new ArrayList<>(); - this.encoding = encoding; - this.setSession(); - params = this.handleParams(fileUrl, params); - this.data = new MissionData(fileUrl, goalPath, rename, suffix, fileExists, split, params, 0L); - this.method = this.data.params.get("data") != null || this.data.params.get("json") != null ? "post" : "get"; - } - - /** - * @param id 任务id - * @param downloadKit 所属DownloadKit对象 - * @param fileUrl 文件网址 - * @param goalPath 保存文件夹路径 - * @param rename 重命名 - * @param suffix 重命名后缀名 - * @param fileExists 存在同名文件处理方式 - * @param split 是否分块下载 - * @param encoding 编码格式 - * @param params 连接参数 - */ - public Mission(String id, DownloadKit downloadKit, String fileUrl, Path goalPath, String rename, String suffix, FileMode fileExists, boolean split, String encoding, Map params) { - super(id); - this.downloadKit = downloadKit; - this.size = null; - this.tasks = new ArrayList<>(); - this.encoding = encoding; - this.setSession(); - params = this.handleParams(fileUrl, params); - this.data = new MissionData(fileUrl, goalPath, rename, suffix, fileExists, split, params, 0L); - this.method = this.data.params.get("data") != null || this.data.params.get("json") != null ? "post" : "get"; - } - - @Override - public String toString() { - return ""; - } - - /** - * @return 返回任务数据 - */ - @Override - public MissionData data() { - return this.data; - } - - /** - * @return 返回文件保存路径 - */ - public String path() { - return this.path; - } - - /** - * @return 返回记录器对象 - */ - public ByteRecorder recorder() { - if (this.recorder == null) { - this.recorder = new ByteRecorder("", 100); - this.recorder.showMsg = false; - } - return this.recorder; - } - - /** - * @return 返回下载进度百分比 - */ - public Float rate() { - if (this.size == null) return null; - int c = 0; - for (Task task : this.tasks) c += task.downloadedSize; - return new BigDecimal(c * 100).divide(new BigDecimal(this.size), 2, RoundingMode.FLOOR).floatValue(); - } - - /** - * 取消该任务,停止未下载完的task - */ - public void cancel() { - this._breakMission("canceled", "已取消"); - } - - /** - * @return 删除下载的文件 - */ - public boolean delFile() { - if (this.path != null && Paths.get(this.path).toFile().exists()) { - try { - return Paths.get(this.path).toFile().delete(); - } catch (Exception ignored) { - - } - } - return false; - } - - /** - * 等待当前任务完成 - * - * @return 任务结果和信息组成的数组 - */ - public String[] waits() { - return wait(true); - } - - /** - * 等待当前任务完成 - * - * @param show 是否显示下载进度 - * @return 任务结果和信息组成的数组 - */ - public String[] wait(boolean show) { - return wait(show, 0); - } - - /** - * 等待当前任务完成 - * - * @param timeout 超时时间 - * @return 任务结果和信息组成的数组 - */ - public String[] wait(double timeout) { - return wait(true, 0); - } - - /** - * 等待当前任务完成 - * - * @param show 是否显示下载进度 - * @param timeout 超时时间 - * @return 任务结果和信息组成的数组 - */ - public String[] wait(boolean show, double timeout) { - if (show) { - System.out.println("url:" + this.data().url); - long t2 = System.currentTimeMillis(); - while (this.fileName == null && System.currentTimeMillis() - t2 < 4000) { - try { - Thread.sleep(10); - } catch (InterruptedException e) { - throw new RuntimeException(e); - } - } - System.out.println("file:" + this.fileName); - System.out.println("filePath:" + this.fileName); - if (this.size == null) System.out.println("Unknown size"); - } - long t1 = System.currentTimeMillis(); - while (!this.isDone() && (System.currentTimeMillis() - t1 < timeout * 1000 || timeout == 0)) { - if (show && this.size != null) { - try { - long rate = Files.size(Paths.get(this.path())); - System.out.println("\r" + new BigDecimal(rate * 100).divide(new BigDecimal(this.size), 2, RoundingMode.FLOOR) + "%"); - } catch (IOException ignored) { - - } - } - try { - Thread.sleep(10); - } catch (InterruptedException e) { - throw new RuntimeException(e); - } - } - if (show) { - String result = this.result.trim().toLowerCase(); - switch (result) { - case "false": - System.out.println("下载失败 " + this.info); - break; - case "success": - System.out.println("100%"); - System.out.println("下载成功 " + this.info); - break; - case "skipped": - System.out.println("已跳过 " + this.info); - break; - } - } - return new String[]{super.result, super.info}; - } - - /** - * 复制Session对象,并设置cookies - */ - private void setSession() { - OkHttpClient.Builder builder = this.downloadKit.session().newBuilder(); - CaseInsensitiveMap headers = new CaseInsensitiveMap<>(); - /* - * 使用拦截器去获取请求头参数 - */ - builder.addInterceptor(new Interceptor() { - @NotNull - @Override - public Response intercept(@NotNull Interceptor.Chain chain) throws IOException { - Request request = chain.request(); - Headers headers1 = request.headers(); - for (Pair extends String, ? extends String> pair : headers1) - headers.put(String.valueOf(pair), headers1.get(String.valueOf(pair))); - return chain.proceed(request.newBuilder().headers(Headers.of()).build()); - } - }); - if (this.downloadKit.getPage() != null) { - Utils.setSessionCookies(builder, this.downloadKit.getPage().cookies()); - try { - Field header; - header = this.downloadKit.getPage().getClass().getField("headers"); - header.setAccessible(true); - Object o = header.get(this.downloadKit.getPage()); - if (o instanceof Map) { - Map, ?> map = (Map, ?>) o; - // 检查泛型参数是否为 - if (map.keySet().stream().allMatch(key -> key instanceof String) && map.values().stream().allMatch(Objects::nonNull)) - map.forEach((a, b) -> headers.put((String) a, b)); - } - } catch (NoSuchFieldException | IllegalAccessException ignored) { - } - headers.put("User-Agent", this.downloadKit.getPage().userAgent()); - } - this.session = builder.build(); - this.headers = headers; - - } - - - /** - * 处理接收到的参数 - * - * @param url 要访问的url - * @param params 传入的参数map - * @return 处理后的参数map - */ - private Map handleParams(String url, Map params) { - if (!params.containsKey("timeout")) params.put("timeout", this.downloadKit.timeout()); - Map headers = params.containsKey("headers") ? new CaseInsensitiveMap<>(JSON.parseObject(params.get("headers").toString())) : new CaseInsensitiveMap<>(); - URI uri = URI.create(url); - String hostName = uri.getHost(); - String scheme = uri.getScheme(); - if (!(headers.containsKey("Referer") || this.headers.containsKey("Referer"))) - headers.put("Referer", this.downloadKit.getPage() != null && this.downloadKit.getPage().url() != null ? this.downloadKit.getPage().url() : scheme + "://" + hostName); - if (!(headers.containsKey("Host") || this.headers.containsKey("Host"))) headers.put("Host", hostName); - params.put("headers", headers); - return params; - - } - - /** - * 设置文件保存路径 - */ - public void _setPath(Object path) { - Path path1; - if (path instanceof Path) path1 = (Path) path; - else if (path instanceof String) { - path1 = Paths.get((String) path); - } else throw new IllegalArgumentException("path只能是String或者Path"); - this.fileName = path1.toAbsolutePath().getFileName().toString(); - this.path = path1.toAbsolutePath().toString(); - this.recorder.set().path(path1); - } - - /** - * 设置一个任务为done状态 - * - * @param result 结果:'success'、'skipped'、'canceled'、False、None - * @param info 任务信息 - */ - public void _setDone(String result, String info) { - switch (result) { - case "skipped": - this.setStates(result, info, Mission.DONE); - break; - case "canceled": - case "false": - this.recorder.clear(); - this.setStates(result, info, Mission.DONE); - break; - case "success": - this.recorder.record(); - try { - if (this.size != null && Files.size(Paths.get(this.path)) < this.size) { - this.delFile(); - this.setStates("false", "下载失败", Mission.DONE); - } else { - this.setStates("success", info, Mission.DONE); - - } - } catch (IOException e) { - throw new RuntimeException(e); - } - } - this.downloadKit._whenMissionDone(this); - } - - /** - * 当一个task完成时调用 - * - * @param isSuccess 该task是否成功 - * @param info 该task传入的信息 - */ - protected void aTaskDone(boolean isSuccess, String info) { - if (this.isDone()) return; - if (!isSuccess) this._breakMission("false", info); - if (++this.doneTasksCount == this.tasksCount) this._setDone("success", info); - } - - /** - * 中止该任务,停止未下载完的task - * - * @param result 结果:'success'、'skipped'、'canceled'、false、None - * @param info 任务信息 - */ - public void _breakMission(String result, String info) { - if (this.isDone()) return; - this.tasks.stream().filter(task -> !task.isDone()).forEach(task -> task.setStates(result, info, "cancel")); - this.tasks.stream().filter(task -> !task.isDone()).forEach(task -> { - try { - Thread.sleep(300); - } catch (InterruptedException e) { - throw new RuntimeException(e); - } - }); - - this._setDone(result, info); - this.delFile(); - } -} diff --git a/java/src/main/java/com/ll/DownloadKit/mission/MissionData.java b/java/src/main/java/com/ll/DownloadKit/mission/MissionData.java deleted file mode 100644 index 20dc3f7..0000000 --- a/java/src/main/java/com/ll/DownloadKit/mission/MissionData.java +++ /dev/null @@ -1,83 +0,0 @@ -package com.ll.DownloadKit.mission; - -import com.ll.DownloadKit.FileMode; -import lombok.Getter; -import lombok.Setter; - -import java.io.UnsupportedEncodingException; -import java.net.URLEncoder; -import java.nio.file.Path; -import java.util.Map; - -/** - * 保存任务数据的对象 - * - * @author 陆 - * @address click - */ -public class MissionData { - /** - * 下载文件url - */ - @Getter - - protected String url; - /** - * 保存文件夹 - */ - @Getter - protected String goalPath; - /** - * 文件重命名 - */ - @Getter - protected String rename; - /** - * 文件重命名后缀名 - */ - @Getter - protected String suffix; - /** - * 存在重名文件时处理方式 - */ - @Getter - protected FileMode fileExists; - /** - * 是否允许分块下载 - */ - @Getter - protected boolean split; - /** - * requests其它参数 - */ - @Getter - protected Map params; - /** - * 文件存储偏移量 - */ - @Setter - @Getter - protected Long offset; - - public MissionData(String url, String goalPath, String rename, String suffix, FileMode fileExists, boolean split, Map params, Long offset) { - if (url != null) { - //版本兼容 - try { - this.url = URLEncoder.encode(url, "utf-8").replaceAll("\\+", "%20").replaceAll("%21", "!").replaceAll("%27", "'").replaceAll("%28", "(").replaceAll("%29", ")").replaceAll("%7E", "~"); - } catch (UnsupportedEncodingException e) { - throw new RuntimeException(e); - } - } - this.goalPath = goalPath; - this.rename = rename; - this.suffix = suffix; - this.fileExists = fileExists; - this.split = split; - this.params = params; - this.offset = offset; - } - - public MissionData(String url, Path goalPath, String rename, String suffix, FileMode fileExists, boolean split, Map params, Long offset) { - this(url, goalPath.toAbsolutePath().toString(),rename,suffix,fileExists,split,params,offset); - } -} diff --git a/java/src/main/java/com/ll/DownloadKit/mission/Task.java b/java/src/main/java/com/ll/DownloadKit/mission/Task.java deleted file mode 100644 index 3623858..0000000 --- a/java/src/main/java/com/ll/DownloadKit/mission/Task.java +++ /dev/null @@ -1,104 +0,0 @@ -package com.ll.DownloadKit.mission; - -import lombok.Getter; - -import java.math.BigDecimal; -import java.math.RoundingMode; -import java.util.List; -import java.util.Objects; - -/** - * 子任务类 - * - * @author 陆 - * @address click - */ -public class Task extends BaseTask { - - @Getter - private Mission mission; - @Getter - List range; - protected int downloadedSize; - private Long size; - - /** - * @param id 任务id - */ - public Task(Mission mission, List range, String id, long size) { - super(id); - this.mission = mission; - this.range = range; - this.size = size; - this.downloadedSize = 0; - } - - /** - * @return 返回父任务id - */ - public String mid() { - return this.mission.id(); - } - - /** - * @return 返回任务数据对象 - */ - public MissionData data() { - return this.mission.data(); - } - - /** - * @return 返回文件保存路径 - */ - - public String path() { - return this.mission.path(); - } - - - /** - * @return 返回文件名 - */ - public String fileName() { - return this.mission.fileName; - } - - /** - * @return 返回下载进度百分比 - */ - - public Float rate() { - return this.size == null ? null : new BigDecimal(this.downloadedSize * 100).divide(new BigDecimal(this.size), 2, RoundingMode.FLOOR).floatValue(); - } - - - public void addData(byte[] data) { - addData(data, null); - } - - public void addData(byte[] data, Long seek) { - this.downloadedSize += data.length; - this.mission.recorder().addData(data, seek); - - } - - /** - * 清除以接收但未写入硬盘的缓存 - */ - public void clearCache() { - this.mission.recorder().clear(); - } - - /** - * 设置一个子任务为done状态 - * - * @param result 结果:'success'、'skipped'、'canceled'、'false'、'null' - * @param info 任务信息 - */ - public void _setDone(String result, String info) { - this.setStates(result, info, Task.DONE); - this.mission.aTaskDone(!Objects.equals(result, "false"), info); - } - - -} diff --git a/java/src/main/java/com/ll/DrissonPage/base/BaseElement.java b/java/src/main/java/com/ll/DrissonPage/base/BaseElement.java deleted file mode 100644 index 4f4ea54..0000000 --- a/java/src/main/java/com/ll/DrissonPage/base/BaseElement.java +++ /dev/null @@ -1,242 +0,0 @@ -package com.ll.DrissonPage.base; - -import com.ll.DrissonPage.error.extend.ElementNotFoundError; -import com.ll.DrissonPage.functions.Settings; -import lombok.Getter; - -import java.util.ArrayList; -import java.util.HashMap; -import java.util.List; -import java.util.Map; - -/** - * 各元素类的基类 - * - * @author 陆 - * @address click - */ -@Getter -public abstract class BaseElement, T extends BaseElement, ?>> extends BaseParser { - private final P owner; - - public BaseElement(P page) { - this.owner = page; - this.setType("BaseElement"); - } - - /** - * @return 返回元素标签名 - */ - public abstract String tag(); - - /** - * 返回上面某一级父元素 用查询语法定位 - * - * @param by 查询选择器 - * @return 上级元素对象 - */ - public T parent(By by) { - return parent(by, 1); - } - - /** - * 返回上面某一级父元素 用查询语法定位 - * - * @param by 查询选择器 - * @param index 选择第几个结果 - * @return 上级元素对象 - */ - public abstract T parent(By by, Integer index); - - /** - * 获取上级父元素 - * - * @return 上级元素对象 - */ - public T parent() { - return parent(1); - } - - /** - * 返回上面某一级父元素,指定层数 - * - * @param level 第几级父元素 - * @return 上级元素对象 - */ - public abstract T parent(Integer level); - - /** - * 获取指定定位的第一个父元素 - * - * @param loc 定位语法 - * @return 上级元素对象 - */ - public T parent(String loc) { - return parent(loc, 1); - } - - /** - * 返回上面某一级父元素 用查询语法定位 - * - * @param loc 定位符 - * @param index 选择第几个结果 - * @return 上级元素对象 - */ - public abstract T parent(String loc, Integer index); - - - /** - * 返回当前元素后面一个符合条件的同级元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param by 用于筛选的查询语法 - * @return 兄弟元素或节点文本 - */ - public T next(By by) { - return next(by, 1); - } - - /** - * 返回当前元素后面一个符合条件的同级元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param by 用于筛选的查询语法 - * @param index 第几个查询结果,0开始 - * @return 兄弟元素或节点文本 - */ - public T next(By by, Integer index) { - return next(by, index, null); - } - - /** - * 返回当前元素后面一个符合条件的同级元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param by 用于筛选的查询语法 - * @param index 第几个查询结果,0开始 - * @param timeout 查找节点的超时时间(秒) - * @return 兄弟元素或节点文本 - */ - public T next(By by, Integer index, Double timeout) { - return next(by, index, timeout, true); - } - - /** - * 返回当前元素后面一个符合条件的同级元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param by 用于筛选的查询语法 - * @param index 第几个查询结果,0开始 - * @param timeout 查找节点的超时时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 兄弟元素或节点文本 - */ - public abstract T next(By by, Integer index, Double timeout, Boolean eleOnly); - - /** - * 返回当前元素后面一个符合条件的同级元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @return 兄弟元素或节点文本 - */ - public T next() { - return next(""); - } - - /** - * 返回当前元素后面一个符合条件的同级元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param loc 用于筛选的查询语法 - * @return 兄弟元素或节点文本 - */ - public T next(String loc) { - return next(loc.isEmpty() ? null : loc, 1); - } - - /** - * 返回当前元素后面一个符合条件的同级元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param loc 用于筛选的查询语法 - * @param index 第几个查询结果,0开始 - * @return 兄弟元素或节点文本 - */ - public T next(String loc, Integer index) { - return next(loc, index, null); - } - - /** - * 返回当前元素后面一个符合条件的同级元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param loc 用于筛选的查询语法 - * @param index 第几个查询结果,0开始 - * @param timeout 查找节点的超时时间(秒) - * @return 兄弟元素或节点文本 - */ - public T next(String loc, Integer index, Double timeout) { - return next(loc, index, timeout, true); - } - - /** - * 返回当前元素后面一个符合条件的同级元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param loc 用于筛选的查询语法 - * @param index 第几个查询结果,0开始 - * @param timeout 查找节点的超时时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 兄弟元素或节点文本 - */ - public abstract T next(String loc, Integer index, Double timeout, Boolean eleOnly); - - /** - * 返回当前元素后面一个符合条件的同级元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param index 第几个查询结果,0开始 - * @return 兄弟元素或节点文本 - */ - - public T next(Integer index) { - return next(index, null); - } - - /** - * 返回当前元素后面一个符合条件的同级元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param index 第几个查询结果,0开始 - * @param timeout 查找节点的超时时间(秒) - * @return 兄弟元素或节点文本 - */ - public T next(Integer index, Double timeout) { - return next(index, timeout, true); - } - - /** - * 返回当前元素后面一个符合条件的同级元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param index 第几个查询结果,0开始 - * @param timeout 查找节点的超时时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 兄弟元素或节点文本 - */ - public abstract T next(Integer index, Double timeout, Boolean eleOnly); - - - @Override - public List _ele(By by, Double timeout, Integer index, Boolean relative, Boolean raiseErr, String method) { - return __ele(this.findElements(by, timeout, index, relative, raiseErr), by.getValue(), index, raiseErr, method); - } - - @Override - public List _ele(String str, Double timeout, Integer index, Boolean relative, Boolean raiseErr, String method) { - return __ele(this.findElements(str, timeout, index, relative, raiseErr), str, index, raiseErr, method); - } - - private List __ele(List elements, String str, Integer index, Boolean raiseErr, String method) { - //如果index为空则说明是查找多元素,如果不为空,则说明是单元素查找,直接判断是否为空,如果不为空则说明单元素找到了 - if (index == null || !elements.isEmpty()) return elements; - //如果是单元素,是否抛出异常 - if (Settings.raiseWhenEleNotFound || raiseErr != null && raiseErr) { - Map locOrStr = new HashMap<>(); - locOrStr.put("loc_or_str", str); - locOrStr.put("index", index); - throw new ElementNotFoundError(null, method, locOrStr); - } - //如果不抛出异常则直接创建个空的 - return new ArrayList<>(); - } - -} diff --git a/java/src/main/java/com/ll/DrissonPage/base/BasePage.java b/java/src/main/java/com/ll/DrissonPage/base/BasePage.java deleted file mode 100644 index f63d502..0000000 --- a/java/src/main/java/com/ll/DrissonPage/base/BasePage.java +++ /dev/null @@ -1,300 +0,0 @@ -package com.ll.DrissonPage.base; - -import com.alibaba.fastjson.JSONObject; -import com.ll.DownloadKit.DownloadKit; -import com.ll.DrissonPage.error.extend.ElementNotFoundError; -import com.ll.DrissonPage.functions.Settings; -import lombok.Getter; -import lombok.Setter; -import okhttp3.Cookie; - -import java.nio.file.InvalidPathException; -import java.nio.file.Path; -import java.nio.file.Paths; -import java.util.ArrayList; -import java.util.HashMap; -import java.util.List; -import java.util.Map; - -/** - * 页面类的基类 - * - * @author 陆 - * @address click - */ -//页面类的基类 -public abstract class BasePage> extends BaseParser { - protected String url = null; - /** - * 查找元素时等待的秒数 - */ - private Double timeout = 10.0; - /** - * 当前访问的url有效性 - */ - @Setter - private Boolean urlAvailable = null; - /** - * 重试次数 - */ - @Setter - @Getter - private Integer retryTimes = 3; - /** - * 重试间隔 - */ - @Setter - @Getter - private Double retryInterval = 2.0; - /** - * 默认下载路径 - */ - @Setter - private String downloadPath = null; - /** - * 下载器对象 - */ - @Getter - private DownloadKit downloadKit = null; - /** - * ele返回值 - */ - @Setter - private Boolean noneEleReturnValue = Boolean.FALSE; - @Setter - private Object noneEleValue = null; - - public BasePage() { - this.setType("BasePage"); - } - - /** - * 返回网页的标题title - */ - public String title() { - T title = this._ele("xpath://title", null, null, null, false, "title").get(0); - return title instanceof DrissionElement ? ((DrissionElement, ?>) title).text() : null; - } - - /** - * 返回查找元素时等待的秒数 - */ - public Double timeout() { - return this.timeout; - } - - /** - * 设置查找元素时等待的秒数 - * - * @param timeout 秒 - */ - public void setTimeout(Double timeout) { - if (timeout != null && timeout >= 0) this.timeout = timeout; - } - - - /** - * 连接前的准备 - * - * @param url 要访问的url - * @param retry 重试次数 - * @param interval 重试间隔 - * @return 重试次数和间隔 - */ - protected BeforeConnect beforeConnect(String url, Integer retry, Double interval) { - boolean isFile = false; - - try { - if (Paths.get(url).toFile().exists() || (!url.contains("://") && !url.contains(":\\\\"))) { - Path p = Paths.get(url); - if (p.toFile().exists()) { - url = p.toAbsolutePath().toString(); - isFile = true; - } - } - } catch (InvalidPathException ignored) { - - } - this.url = url; - retry = retry == null ? this.getRetryTimes() : retry; - interval = interval == null ? this.getRetryInterval() : interval; - return new BeforeConnect(retry, interval, isFile); - } - - /** - * @return 返回当前访问的url有效性 - */ - public Boolean urlAvailable() { - return this.urlAvailable; - } - - /** - * @return 返回默认下载路径 - */ - public String downloadPath() { - return this.downloadPath; - } - - /** - * 返回下载器对象 - */ - public DownloadKit download() { - if (this.downloadKit == null) this.downloadKit = new DownloadKit(this.downloadPath, null, null, this); - return this.downloadKit; - } - - //---------------------------------------------------------------------------------------------------------------------- - - /** - * @return 返回当前访问url - */ - public abstract String url(); - /** - * @return 用于被WebPage覆盖 - */ - protected String browserUrl() { - return this.url(); - } - - public abstract JSONObject json(); - - /** - * @return 返回user agent - */ - public abstract String userAgent(); - - /** - * 返回cookies - */ - public List cookies() { - return cookies(false); - } - - /** - * 返回cookies - * - * @param asMap 为True时返回由{name: value}键值对组成的map,为false时返回list且allInfo无效 - */ - public List cookies(boolean asMap) { - return cookies(asMap, false); - } - - /** - * 返回cookies - * - * @param asMap 为True时返回由{name: value}键值对组成的map,为false时返回list且allInfo无效 - * @param allDomains 是否返回所有域的cookies - */ - public List cookies(boolean asMap, boolean allDomains) { - return cookies(asMap, allDomains, false); - } - - /** - * 返回cookies - * - * @param asMap 为True时返回由{name: value}键值对组成的map,为false时返回list且allInfo无效 - * @param allDomains 是否返回所有域的cookies - * @param allInfo 是否返回所有信息,为False时只返回name、value、domain - */ - public abstract List cookies(boolean asMap, boolean allDomains, boolean allInfo); - - public Boolean get(String url) { - return get(url, false); - } - - - public Boolean get(String url, boolean showErrMsg) { - return get(url, showErrMsg, retryTimes, retryInterval, timeout); - } - - public Boolean get(String url, boolean showErrMsg, Integer retry, Double interval, Double timeout) { - return get(url, showErrMsg, retry, interval, timeout, null); - } - - /** - * 用get请求跳转到url,可输入文件路径 - * - * @param url 目标url - * @param showErrMsg 是否显示和抛出异常 - * @param retry 重试次数,为null时使用页面对象retryTimes属性值 - * @param interval 重试间隔(秒),为null时使用页面对象retryInterval属性值 - * @param timeout 连接超时时间(秒),为null时使用页面对象timeouts.pageLoad属性值 - * @param params 连接参数 s模式专用 - * @return url是否可用 - */ - public abstract Boolean get(String url, boolean showErrMsg, Integer retry, Double interval, Double timeout, Map params); - - /** - * @param by 查询元素 - * @param timeout 查找超时时间(秒) - * @param index 获取第几个,从0开始,可传入负数获取倒数第几个 如果是null则是返回全部 - * @param relative WebPage用的表示是否相对定位的参数 - * @param raiseErr 找不到元素是是否抛出异常,为null时根据全局设置 - * @param method 方法名称 - * @return 元素对象组成的列表 - */ - @Override - public List _ele(By by, Double timeout, Integer index, Boolean relative, Boolean raiseErr, String method) { - Map map = new HashMap<>(); - map.put("str", ""); - map.put("index", index); - if (by == null) throw new ElementNotFoundError(null, method, map); - List elements = this.findElements(by, timeout, index, relative, raiseErr); - //如果index为空则说明是查找多元素,如果不为空,则说明是单元素查找,直接判断是否为空,如果不为空则说明单元素找到了 - if (index == null || !elements.isEmpty()) return elements; - //如果是单元素,是否抛出异常 - if (Settings.raiseWhenEleNotFound || raiseErr != null && raiseErr) - throw new ElementNotFoundError(null, method, map); - //如果不抛出异常则直接创建个空的 - return new ArrayList<>(); - } - - /** - * @param loc 定位符 - * @param timeout 查找超时时间(秒) - * @param index 获取第几个,从0开始,可传入负数获取倒数第几个 如果是null则是返回全部 - * @param relative WebPage用的表示是否相对定位的参数 - * @param raiseErr 找不到元素是是否抛出异常,为null时根据全局设置 - * @param method 方法名称 - * @return 元素对象组成的列表 - */ - @Override - public List _ele(String loc, Double timeout, Integer index, Boolean relative, Boolean raiseErr, String method) { - Map map = new HashMap<>(); - map.put("str", ""); - map.put("index", index); - if (loc == null) throw new ElementNotFoundError(null, method, map); - List elements = this.findElements(loc, timeout, index, relative, raiseErr); - //如果index为空则说明是查找多元素,如果不为空,则说明是单元素查找,直接判断是否为空,如果不为空则说明单元素找到了 - if (index == null || !elements.isEmpty()) return elements; - //如果是单元素,是否抛出异常 - if (Settings.raiseWhenEleNotFound || raiseErr != null && raiseErr) - throw new ElementNotFoundError(null, method, map); - //如果不抛出异常则直接创建个空的 - return new ArrayList<>(); - } - /** - * 执行元素查找 - * - * @param by 查询元素 - * @param timeout 查找超时时间(秒) - * @param index 获取第几个,从1开始,可传入负数获取倒数第几个 如果是null则是返回全部 - * @param relative WebPage用的表示是否相对定位的参数 - * @param raiseErr 找不到元素是是否抛出异常,为null时根据全局设置 - * @return 元素对象组成的列表 - */ - protected abstract List findElements(By by, Double timeout, Integer index, Boolean relative, Boolean raiseErr); - - - /** - * 执行元素查找 - * - * @param loc 定位符 - * @param timeout 查找超时时间(秒) - * @param index 获取第几个,从1开始,可传入负数获取倒数第几个 如果是null则是返回全部 - * @param relative WebPage用的表示是否相对定位的参数 - * @param raiseErr 找不到元素是是否抛出异常,为null时根据全局设置 - * @return 元素对象组成的列表 - */ - protected abstract List findElements(String loc, Double timeout, Integer index, Boolean relative, Boolean raiseErr); -} diff --git a/java/src/main/java/com/ll/DrissonPage/base/BaseParser.java b/java/src/main/java/com/ll/DrissonPage/base/BaseParser.java deleted file mode 100644 index 6521aa7..0000000 --- a/java/src/main/java/com/ll/DrissonPage/base/BaseParser.java +++ /dev/null @@ -1,235 +0,0 @@ -package com.ll.DrissonPage.base; - -import com.ll.DrissonPage.element.SessionElement; -import com.ll.DrissonPage.error.extend.ElementNotFoundError; -import lombok.Getter; -import lombok.Setter; - -import java.util.List; -import java.util.Map; - -/** - * 所有页面、元素类的基类 - * - * @author 陆 - * @address click - */ - -@Getter -public abstract class BaseParser> { - @Setter - private String type; - - - /** - * 返回当前元素下级符合条件的一个元素、属性或节点文本 - * - * @param by 查询元素 - */ - public T ele(By by) { - return ele(by, 1); - } - - /** - * 返回当前元素下级符合条件的一个元素、属性或节点文本 - * - * @param by 查询元素 - * @param index 获取第几个元素,下标从1开始可传入负数获取倒数第几个 - */ - public T ele(By by, int index) { - return ele(by, index, null); - } - - /** - * 返回当前元素下级符合条件的一个元素、属性或节点文本 - * - * @param by 查询元素 - * @param timeout 查找元素超时时间(秒),默认与元素所在页面等待时间一致 - */ - public T ele(By by, Double timeout) { - return ele(by, 1, timeout); - } - - /** - * 返回当前元素下级符合条件的一个元素、属性或节点文本 - * - * @param by 查询元素 - * @param index 获取第几个元素,下标从1开始可传入负数获取倒数第几个 - * @param timeout 查找元素超时时间(秒),默认与元素所在页面等待时间一致 - */ - public T ele(By by, int index, Double timeout) { - try { - return this._ele(by, timeout, index, null, null, "ele()").get(0); - } catch (IndexOutOfBoundsException e) { - Map map = new java.util.HashMap<>(); - map.put("by", by); - map.put("index", index); - new ElementNotFoundError("ele()", map).printStackTrace(); - return null; - } - } - - /** - * 获取单个元素 - * - * @param loc 参数,字符串 - */ - public T ele(String loc) { - return ele(loc, 1); - } - - /** - * 获取单个元素 - * - * @param loc 参数,字符串 - * @param index 获取第几个元素,下标从0开始可传入负数获取倒数第几个 - */ - public T ele(String loc, int index) { - return ele(loc, index, null); - } - - - /** - * 返回当前元素下级符合条件的一个元素、属性或节点文本 - * - * @param loc 参数,字符串 - * @param timeout 查找元素超时时间(秒),默认与元素所在页面等待时间一致 - */ - public T ele(String loc, Double timeout) { - return ele(loc, 1, timeout); - } - - /** - * 获取单个元素 - * - * @param loc 参数,字符串 - * @param index 获取第几个元素,下标从0开始可传入负数获取倒数第几个 - * @param timeout 查找元素超时时间(秒),默认与元素所在页面等待时间一致 - */ - public T ele(String loc, int index, Double timeout) { - try { - List ts = this._ele(loc, timeout, index, null, null, "ele()"); - if (ts == null) return null; - return ts.get(0); - } catch (IndexOutOfBoundsException e) { - Map map = new java.util.HashMap<>(); - map.put("loc", loc); - map.put("index", index); - new ElementNotFoundError("ele()", map).printStackTrace(); - return null; - } - } - - - public List eles(By by) { - return eles(by, null); - } - - public List eles(By by, Double timeout) { - return this._ele(by, timeout, null, null, null, null); - } - - public List eles(String loc) { - return eles(loc, null); - } - - public List eles(String loc, Double timeout) { - return this._ele(loc, timeout, null, null, null, null); - } - - /** - * 获取当前页面数据 - */ - public abstract String html(); - - /** - * @param by 查询元素 - * @return 元素对象 - */ - public SessionElement sEle(By by) { - return sEle(by, 1); - } - - /** - * @param by 查询元素 - * @param index 获取第几个,从1开始,可传入负数获取倒数第几个 - * @return 元素对象 - */ - public abstract SessionElement sEle(By by, Integer index); - - /** - * @param loc 定位符 - * @return 元素对象 - */ - public SessionElement sEle(String loc) { - return sEle(loc, 1); - } - - /** - * @param loc 定位符 - * @param index 获取第几个,从1开始,可传入负数获取倒数第几个 - * @return 元素对象 - */ - public abstract SessionElement sEle(String loc, Integer index); - - /** - * @param by 查询元素 - * @return 元素对象组成的列表 - */ - public abstract List sEles(By by); - - /** - * @param loc 定位符 - * @return 元素对象组成的列表 - */ - public abstract List sEles(String loc); - - - /** - * @param by 查询元素 - * @param timeout 查找超时时间(秒) - * @param index 获取第几个,从0开始,可传入负数获取倒数第几个 如果是null则是返回全部 - * @param relative WebPage用的表示是否相对定位的参数 - * @param raiseErr 找不到元素是是否抛出异常,为null时根据全局设置 - * @param method 方法名称 - * @return 元素对象组成的列表 - */ - protected abstract List _ele(By by, Double timeout, Integer index, Boolean relative, Boolean raiseErr, String method); - - /** - * @param loc 定位符 - * @param timeout 查找超时时间(秒) - * @param index 获取第几个,从0开始,可传入负数获取倒数第几个 如果是null则是返回全部 - * @param relative WebPage用的表示是否相对定位的参数 - * @param raiseErr 找不到元素是是否抛出异常,为null时根据全局设置 - * @param method 方法名称 - * @return 元素对象组成的列表 - */ - protected abstract List _ele(String loc, Double timeout, Integer index, Boolean relative, Boolean raiseErr, String method); - - /** - * 执行元素查找 - * - * @param by 查询元素 - * @param timeout 查找超时时间(秒) - * @param index 获取第几个,从0开始,可传入负数获取倒数第几个 如果是null则是返回全部 - * @param relative WebPage用的表示是否相对定位的参数 - * @param raiseErr 找不到元素是是否抛出异常,为null时根据全局设置 - * @return 元素对象组成的列表 - */ - protected abstract List findElements(By by, Double timeout, Integer index, Boolean relative, Boolean raiseErr); - - - /** - * 执行元素查找 - * - * @param loc 定位符 - * @param timeout 查找超时时间(秒) - * @param index 获取第几个,从0开始,可传入负数获取倒数第几个 如果是null则是返回全部 - * @param relative WebPage用的表示是否相对定位的参数 - * @param raiseErr 找不到元素是是否抛出异常,为null时根据全局设置 - * @return 元素对象组成的列表 - */ - protected abstract List findElements(String loc, Double timeout, Integer index, Boolean relative, Boolean raiseErr); - -} \ No newline at end of file diff --git a/java/src/main/java/com/ll/DrissonPage/base/BeforeConnect.java b/java/src/main/java/com/ll/DrissonPage/base/BeforeConnect.java deleted file mode 100644 index 74a59d6..0000000 --- a/java/src/main/java/com/ll/DrissonPage/base/BeforeConnect.java +++ /dev/null @@ -1,16 +0,0 @@ -package com.ll.DrissonPage.base; - -import lombok.AllArgsConstructor; -import lombok.Getter; - -/** - * @author 陆 - * @address click - */ -@AllArgsConstructor -@Getter -public class BeforeConnect { - private Integer retry; - private Double interval; - private boolean isFile; -} diff --git a/java/src/main/java/com/ll/DrissonPage/base/Browser.java b/java/src/main/java/com/ll/DrissonPage/base/Browser.java deleted file mode 100644 index 8a6d5fa..0000000 --- a/java/src/main/java/com/ll/DrissonPage/base/Browser.java +++ /dev/null @@ -1,470 +0,0 @@ -package com.ll.DrissonPage.base; - -import com.alibaba.fastjson.JSON; -import com.alibaba.fastjson.JSONArray; -import com.alibaba.fastjson.JSONObject; -import com.ll.DrissonPage.config.ChromiumOptions; -import com.ll.DrissonPage.error.extend.PageDisconnectedError; -import com.ll.DrissonPage.functions.Tools; -import com.ll.DrissonPage.page.ChromiumPage; -import com.ll.DrissonPage.page.ChromiumBase; -import com.ll.DrissonPage.units.downloader.DownloadManager; -import lombok.Getter; - -import java.io.BufferedReader; -import java.io.File; -import java.io.IOException; -import java.io.InputStreamReader; -import java.nio.file.FileVisitOption; -import java.nio.file.Files; -import java.nio.file.Path; -import java.nio.file.Paths; -import java.util.*; -import java.util.concurrent.ConcurrentHashMap; -import java.util.stream.Collectors; -import java.util.stream.Stream; - -/** - * 浏览器 - * - * @author 陆 - * @address click - */ - -public class Browser implements Occupant { - public static final String __ERROR__ = "error"; - private static final Map BROWSERS = new ConcurrentHashMap<>(); - @Getter - private final ChromiumBase page; - private final Map drivers; - private final Map> allDrivers; - @Getter - private BrowserDriver driver; - @Getter - private String id; - @Getter - private String address; - @Getter - private Map frames; - @Getter - private DownloadManager dlMgr; - /** - * 浏览器进程id - */ - @Getter - private Integer processId; - private boolean connected; - - /** - * 浏览器 - * - * @param address 浏览器地址 - * @param browserId 浏览器id - * @param page ChromiumPage对象 - */ - private Browser(String address, String browserId, ChromiumBase page) { - this.page = page; - this.address = address; - this.driver = BrowserDriver.getInstance(browserId, "browser", address, this); - this.id = browserId; - this.frames = new HashMap<>(); - this.drivers = new HashMap<>(); - this.allDrivers = new HashMap<>(); - this.connected = false; - this.processId = null; - JSONArray processInfoArray = JSON.parseObject(runCdp("SystemInfo.getProcessInfo")).getJSONArray("processInfo"); - if (processInfoArray == null) processInfoArray = new JSONArray(); - for (Object processInfoObject : processInfoArray) { - JSONObject processInfo = JSON.parseObject(processInfoObject.toString()); - if ("browser".equals(processInfo.getString("type"))) { - this.processId = processInfo.getInteger("id"); - break; - } - } - this.runCdp("Target.setDiscoverTargets", Map.of("discover", true)); - driver.setCallback("Target.targetDestroyed", new MyRunnable() { - @Override - public void run() { - onTargetDestroyed(getMessage()); - } - }); - driver.setCallback("Target.targetCreated", new MyRunnable() { - @Override - public void run() { - onTargetCreated(getMessage()); - } - }); - } - - /** - * 单例模式 - * - * @param address 浏览器地址 - * @param browserId 浏览器id - * @param page ChromiumPage对象 - */ - public static Browser getInstance(String address, String browserId, ChromiumBase page) { - return BROWSERS.computeIfAbsent(browserId, key -> new Browser(address, browserId, page)); - } - - /** - * 获取对应tab id的Driver - * - * @param tabId 标签页id - * @return Driver对象 - */ - public Driver getDriver(String tabId) { - return getDriver(tabId, null); - } - - /** - * 获取对应tab id的Driver - * - * @param tabId 标签页id - * @param occupant 使用该驱动的对象 - * @return Driver对象 - */ - public Driver getDriver(String tabId, Occupant occupant) { - Driver driver = Objects.requireNonNullElseGet(drivers.remove(tabId), () -> new Driver(tabId, "page", this.getAddress(), occupant)); - HashSet value = new HashSet<>(); - value.add(driver); - this.allDrivers.put(tabId, value); - return driver; - } - - /** - * 标签页创建时执行 - * - * @param message 回调参数 - */ - private void onTargetCreated(Object message) { - try { - JSONObject jsonObject = JSON.parseObject(message.toString()); - String type = jsonObject.getJSONObject("targetInfo").getString("type"); - if ("page".equals(type) || "webview".equals(type) && !jsonObject.getJSONObject("targetInfo").getString("url").startsWith("devtools://")) { - String targetId = jsonObject.getJSONObject("targetInfo").getString("targetId"); - Driver driver = new Driver(targetId, "page", address); - drivers.put(targetId, driver); - this.allDrivers.computeIfAbsent(targetId, k -> new HashSet<>()).add(driver); - } - } catch (Exception e) { - e.printStackTrace(); - } - } - - /** - * 标签页关闭时执行 - * - * @param message 回调参数 - */ - private void onTargetDestroyed(Object message) { - JSONObject jsonObject = JSON.parseObject(message.toString()); - String tabId = jsonObject.getString("targetId"); - if (this.dlMgr != null) this.dlMgr.clearTabInfo(tabId); - if (frames != null) frames.values().removeIf(value -> value.equals(tabId)); - allDrivers.forEach((a, b) -> { - if (a.equals(tabId)) b.forEach(Driver::stop); - }); - allDrivers.remove(tabId); - drivers.forEach((a, b) -> { - if (a.equals(tabId)) b.stop(); - }); - drivers.remove(tabId); - - } - - /** - * 执行page相关的逻辑 - */ - public void connectToPage() { - if (!connected) { - this.dlMgr = new DownloadManager(this); - connected = true; - } - } - - public String runCdp(String cmd) { - return runCdp(cmd, new HashMap<>()); - } - - /** - * 执行Chrome DevTools Protocol语句 - * - * @param cmd 协议项目 - * @param cmdArgs 参数 - * @return 执行的结果 - */ - public String runCdp(String cmd, Map cmdArgs) { - cmdArgs = new HashMap<>(cmdArgs == null ? new HashMap<>() : cmdArgs); - Object ignore = cmdArgs.remove("_ignore"); - String result = driver.run(cmd, cmdArgs).toString(); - JSONObject result1 = JSONObject.parseObject(result); - if (result1.containsKey(__ERROR__)) Tools.raiseError(result1, ignore); - return result; - } - - /** - * 返回标签页数量 - */ - public int tabsCount() { - JSONArray targetInfos = JSON.parseObject(this.runCdp("Target.getTargets")).getJSONArray("targetInfos"); - return (int) targetInfos.stream().filter(targetInfo -> { - JSONObject jsonObject = JSON.parseObject(targetInfo.toString()); - String type = jsonObject.getString("type"); - String url = jsonObject.getString("url"); - return ("page".equals(type) || "webview".equals(type)) && !url.startsWith("devtools://"); - }).count(); - } - - /** - * 返回所有标签页id组成的列表 - */ - public List tabs() { - JSONArray jsonArray = JSON.parseArray(JSONObject.toJSONString(driver.get("http://" + address + "/json"))); - return jsonArray.stream().filter(targetInfo -> { - JSONObject jsonObject = JSON.parseObject(targetInfo.toString()); - String type = jsonObject.getString("type"); - String url = jsonObject.getString("url"); - return ("page".equals(type) || "webview".equals(type)) && !url.startsWith("devtools://"); - }).map(obj -> ((JSONObject) obj).getString("id")).collect(Collectors.toList()); - } - - /** - * 查找符合条件的tab,返回它们的id组成的列表 - * - * @return tab id或tab列表 - */ - public List findTabs() { - return findTabs(null); - } - - /** - * 查找符合条件的tab,返回它们的id组成的列表 - * - * @param title 要匹配title的文本 - * @return tab id或tab列表 - */ - public List findTabs(String title) { - return findTabs(title, null); - } - - /** - * 查找符合条件的tab,返回它们的id组成的列表 - * - * @param title 要匹配title的文本 - * @param url 要匹配url的文本 - * @return tab id或tab列表 - */ - public List findTabs(String title, String url) { - return findTabs(title, url, null); - } - - /** - * 查找符合条件的tab,返回它们的id组成的列表 - * - * @param title 要匹配title的文本 - * @param url 要匹配url的文本 - * @param tabType tab类型,可用列表输入多个 - * @return tab id或tab列表 - */ - public List findTabs(String title, String url, List tabType) { - return findTabs(title, url, tabType, true); - } - - /** - * 查找符合条件的tab,返回它们的id组成的列表 - * - * @param title 要匹配title的文本 - * @param url 要匹配url的文本 - * @param tabType tab类型,可用列表输入多个 - * @param single 是否返回首个结果的id,为False返回所有信息 - * @return tab id或tab列表 - */ - public List findTabs(String title, String url, List tabType, boolean single) { - Object parse = JSON.parse(JSONObject.toJSONString(this.driver.get("http://" + this.address + "/json"))); - JSONArray tabs; - if (parse instanceof String) { - tabs = new JSONArray(List.of(parse)); - } else if (parse instanceof List || parse instanceof Set || parse instanceof String[]) { - tabs = JSON.parseArray(parse.toString()); - } else { - throw new IllegalArgumentException("tab_type类型不对" + parse.toString()); - } - - List result = tabs.stream().filter(targetInfo -> { - JSONObject jsonObject = JSON.parseObject(targetInfo.toString()); - return (title == null || jsonObject.getString("title").contains(title)) && (url == null || jsonObject.getString("url").contains(url)) && (tabType == null || tabType.contains(jsonObject.getString("type"))); - }).map(tab -> ((JSONObject) tab).getString("id")).collect(Collectors.toList()); - return single ? (result.isEmpty() ? null : List.of(result.get(0))) : result; - } - - /** - * 关闭标签页 - * - * @param tabId 标签页id - */ - public void closeTab(String tabId) { - this.onTargetDestroyed(JSON.toJSONString(Map.of("targetId", tabId))); - this.driver.run("Target.closeTarget", Map.of("targetId", tabId)); - } - - /** - * 停止一个Driver - * - * @param driver Driver对象 - */ - public void stopDiver(Driver driver) { - driver.stop(); - Set set = this.allDrivers.get(driver.getId()); - if (set != null) set.remove(driver); - } - - /** - * 使标签页变为活动状态 - * - * @param tabId 标签页id - */ - public void activateTab(String tabId) { - this.runCdp("Target.activateTarget", Map.of("targetId", tabId)); - } - - /** - * 返回浏览器窗口位置和大小信息 - * - * @return 窗口大小字典 - */ - public JSONObject getWindowBounds() { - return getWindowBounds(null); - } - - /** - * 返回浏览器窗口位置和大小信息 - * - * @param tabId 标签页id 不传入默认使用本身的id - */ - public JSONObject getWindowBounds(String tabId) { - return JSON.parseObject(runCdp("Browser.getWindowForTarget", Map.of("targetId", tabId == null || tabId.isEmpty() ? this.id : tabId))).getJSONObject("bounds"); - } - - /** - * 断开重连 - */ - public void reconnect() { - this.driver.stop(); - BrowserDriver.BROWSERS.remove(this.id); - this.driver = BrowserDriver.getInstance(this.id, "browser", this.address, this); - this.runCdp("Target.setDiscoverTargets", Map.of("discover", true)); - this.driver.setCallback("Target.targetDestroyed", new MyRunnable() { - @Override - public void run() { - onTargetDestroyed(getMessage()); - } - }); - this.driver.setCallback("Target.targetCreated", new MyRunnable() { - @Override - public void run() { - onTargetCreated(getMessage()); - } - }); - } - - /** - * 关闭浏览器 - */ - public void quit() { - quit(5.0); - } - - /** - * 关闭浏览器 - * - * @param timeout 等待浏览器关闭超时时间 - */ - public void quit(double timeout) { - quit(timeout, false); - } - - /** - * 关闭浏览器 - * - * @param timeout 等待浏览器关闭超时时间 - * @param force 是否立刻强制终止进程 - */ - public void quit(double timeout, boolean force) { - List pids = JSON.parseArray(this.runCdp("SystemInfo.getProcessInfo")).stream().map(o -> JSON.parseObject(o.toString()).getInteger("id")).collect(Collectors.toList()); - - - for (Set value : this.allDrivers.values()) for (Driver driver1 : value) driver1.stop(); - - if (force) { - for (Integer pid : pids) - try { - ProcessHandle.allProcesses().filter(process -> process.info().command().isPresent()).filter(process -> process.info().command().map(s -> s.contains(Integer.toString(pid))).orElse(false)).forEach(process -> { - if (process.isAlive()) process.destroy(); - }); - } catch (SecurityException ignored) { - } - - } else { - try { - this.runCdp("Browser.close"); - this.driver.stop(); - } catch (PageDisconnectedError e) { - this.driver.stop(); - return; - } - } - - - if (force) { - String[] ipPort = address.split(":"); - if (!("127.0.0.1".equals(ipPort[0]) || "localhost".equals(ipPort[0]))) { - return; - } - Tools.stopProcessOnPort(Integer.parseInt(ipPort[1])); - return; - } - - if (processId != null) { - String txt = System.getProperty("os.name").toLowerCase().contains("win") ? "tasklist | findstr " + processId : "ps -ef | grep " + processId; - long endTime = (long) (System.currentTimeMillis() + timeout * 1000); - while (System.currentTimeMillis() < endTime) { - try (BufferedReader reader = new BufferedReader(new InputStreamReader(new ProcessBuilder(txt.split("\\s+")).start().getInputStream()))) { - try { - if (!reader.readLine().contains(processId.toString())) { - return; - } - } catch (NullPointerException e) { - // Handle null pointer exception, if needed - } - Thread.sleep(100); - } catch (IOException | InterruptedException e) { - e.printStackTrace(); - } - } - } - } - - @Override - public void onDisconnect() { - this.page.onDisconnect(); - BROWSERS.remove(id); - if (page instanceof ChromiumPage) { - ChromiumOptions chromiumOptions = ((ChromiumPage) page).getChromiumOptions(); - if (chromiumOptions.isAutoPort() && chromiumOptions.getUserDataPath() != null) { - Path path = Paths.get(chromiumOptions.getUserDataPath()); - long endTime = System.currentTimeMillis() + 7000; - while (System.currentTimeMillis() < endTime) { - if (!Files.exists(path)) break; - try (Stream walk = Files.walk(path, FileVisitOption.FOLLOW_LINKS)) { - walk.sorted(Comparator.reverseOrder()).map(Path::toFile).forEach(File::delete); - break; - } catch (IOException e) { - e.printStackTrace(); - } - } - } - } - - } -} \ No newline at end of file diff --git a/java/src/main/java/com/ll/DrissonPage/base/BrowserDriver.java b/java/src/main/java/com/ll/DrissonPage/base/BrowserDriver.java deleted file mode 100644 index e983b7e..0000000 --- a/java/src/main/java/com/ll/DrissonPage/base/BrowserDriver.java +++ /dev/null @@ -1,49 +0,0 @@ -package com.ll.DrissonPage.base; - -import com.alibaba.fastjson.JSON; -import com.ll.DrissonPage.utils.CloseableHttpClientUtils; -import org.apache.http.client.methods.HttpGet; - -import java.util.Map; -import java.util.concurrent.ConcurrentHashMap; - -/** - * 浏览器驱动 - * - * @author 陆 - * @address click - */ - -public class BrowserDriver extends Driver { - protected static final Map BROWSERS = new ConcurrentHashMap<>(); - - - private BrowserDriver(String tabId, String tabType, String address, Occupant occupant) { - super(tabId, tabType, address, occupant); - } - - public static BrowserDriver getInstance(String tabId, String tabType, String address, Occupant occupant) { - - return BROWSERS.computeIfAbsent(tabId, key -> new BrowserDriver(tabId, tabType, address, occupant)); - } - - @Override - public String toString() { - return ""; - } - - /** - * 发送请求 - * - * @param url 请求地址 - * @return 返回值可能是List 或者Map - */ - public Object get(String url) { - HttpGet request = new HttpGet(url); - request.setHeader("Connection", "close"); - String text = CloseableHttpClientUtils.sendRequestJson(request); - return JSON.parse(text); - } - - -} \ No newline at end of file diff --git a/java/src/main/java/com/ll/DrissonPage/base/By.java b/java/src/main/java/com/ll/DrissonPage/base/By.java deleted file mode 100644 index 9b9c736..0000000 --- a/java/src/main/java/com/ll/DrissonPage/base/By.java +++ /dev/null @@ -1,91 +0,0 @@ -package com.ll.DrissonPage.base; - -import lombok.Getter; -import lombok.Setter; - -import java.util.Objects; - -/** - * @author 陆 - * @address click - */ - -@Getter -@Setter -public class By { - private BySelect name; - private String value; - - private By(BySelect name, String value) { - - this.name = name; - this.value = value; - } - - public static By NULL() { - return null; - } - - public static By id(String value) { - return new By(BySelect.ID, Objects.requireNonNullElse(value, "Cannot find elements when id is null.")); - } - - public static By className(String value) { - return new By(BySelect.CLASS_NAME, Objects.requireNonNullElse(value, "Cannot find elements when the class name expression is null.")); - } - - public static By tag(String value) { - return new By(BySelect.TAG_NAME, Objects.requireNonNullElse(value, "Cannot find elements when the tag name is null.")); - } - - public static By name(String value) { - return new By(BySelect.NAME, Objects.requireNonNullElse(value, "Cannot find elements when name text is null.")); - } - - public static By css(String value) { - return new By(BySelect.CSS_SELECTOR, Objects.requireNonNullElse(value, "Cannot find elements when the css selector is null.")); - } - - public static By xpath(String value) { - return new By(BySelect.XPATH, Objects.requireNonNullElse(value, "Cannot find elements when the XPath is null.")); - } - - public static By linkText(String value) { - return new By(BySelect.LINK_TEXT, Objects.requireNonNullElse(value, "Cannot find elements when the link text is null.")); - } - - public static By partialLinkText(String value) { - return new By(BySelect.PARTIAL_LINK_TEXT, Objects.requireNonNullElse(value, "Cannot find elements when the partial link text is null.")); - } - - public static By text(String value) { - return new By(BySelect.TEXT, Objects.requireNonNullElse(value, "Cannot find elements when the text is null.")); - } - - public static By partialText(String value) { - return new By(BySelect.PARTIAL_TEXT, Objects.requireNonNullElse(value, "Cannot find elements when the partial text is null.")); - } - - @Override - public boolean equals(Object o) { - if (this == o) return true; - if (!(o instanceof By)) return false; - By by = (By) o; - return Objects.equals(getName().getName(), by.getName().getName()) && Objects.equals(getValue(), by.getValue()); - } - - @Override - public int hashCode() { - return Objects.hash(getName(), getValue()); - } - - @Override - public String toString() { - return "By{" + - "name=" + name.getName() + - ", value='" + value + '\'' + - '}'; - } - - -} diff --git a/java/src/main/java/com/ll/DrissonPage/base/BySelect.java b/java/src/main/java/com/ll/DrissonPage/base/BySelect.java deleted file mode 100644 index 5531afb..0000000 --- a/java/src/main/java/com/ll/DrissonPage/base/BySelect.java +++ /dev/null @@ -1,18 +0,0 @@ -package com.ll.DrissonPage.base; - -import lombok.Getter; - -/** - * @author 陆 - * @address click - */ -@Getter -public enum BySelect { - NAME("name"), ID("id"), CLASS_NAME("class name"), TAG_NAME("tag name"), CSS_SELECTOR("css selector"), - XPATH("xpath"), LINK_TEXT("link text"), PARTIAL_LINK_TEXT("partial link text"), TEXT("text"), PARTIAL_TEXT("partial text"); - private final String name; - - BySelect(String name) { - this.name = name; - } -} diff --git a/java/src/main/java/com/ll/DrissonPage/base/DrissionElement.java b/java/src/main/java/com/ll/DrissonPage/base/DrissionElement.java deleted file mode 100644 index ca607d4..0000000 --- a/java/src/main/java/com/ll/DrissonPage/base/DrissionElement.java +++ /dev/null @@ -1,1270 +0,0 @@ -package com.ll.DrissonPage.base; - -import com.ll.DrissonPage.error.extend.ElementNotFoundError; -import com.ll.DrissonPage.functions.Locator; -import com.ll.DrissonPage.functions.Settings; -import com.ll.DrissonPage.functions.Web; - -import java.util.ArrayList; -import java.util.List; -import java.util.Map; -import java.util.regex.Pattern; -import java.util.stream.Collectors; - -/** - * ChromiumElement 和 SessionElement的基类 但不是ShadowRoot的基类 - * - * @author 陆 - * @address click - */ -public abstract class DrissionElement, T extends DrissionElement, ?>> extends BaseElement { - - public DrissionElement(P page) { - super(page); - this.setType("DrissionElement"); - } - - /** - * 返回href或者src绝对的url - */ - public String link() { - String href = this.attr("href"); - return href == null ? this.attr("src") : href; - } - - /** - * 返回css path路径 - */ - public String cssPath() { - return this.getElePath(ElePathMode.CSS); - } - - /** - * 返回xpath路径 - */ - public String xpath() { - return this.getElePath(ElePathMode.XPATH); - } - - /** - * 返回元素注释文本组成的列表 - */ - public List comments() { - return this.eles("xpath:.//comment()"); - } - - /** - * 返回元素内所有直接子节点的文本,包括元素和文本节点 - * - * @return 文本列表 - */ - public List texts() { - return this.texts(false); - } - - /** - * 返回元素内所有直接子节点的文本,包括元素和文本节点 - * - * @param textNodeOnly 是否只返回文本节点 - * @return 文本列表 - */ - public List texts(boolean textNodeOnly) { - List texts = new ArrayList<>(); - if (textNodeOnly) { - this.eles("xpath:/text()").forEach(a -> texts.add(a.text())); - } else { - this.eles("xpath:/text() | *").forEach(a -> texts.add(a.text())); - } - return texts.stream().filter(text -> text != null && !Pattern.compile("[\r\n\t ]").matcher(text).replaceAll("").isEmpty()).map(text -> Web.formatHtml(text.trim().replaceAll("[\r\n]", ""))).collect(Collectors.toList()); - } - - /** - * 返回上面某一级父元素 用查询语法定位 - * - * @param by 查询选择器 - * @param index 选择第几个结果 - * @return 上级元素对象 - */ - @Override - public T parent(By by, Integer index) { - by = Locator.getLoc(by, true, false); - if (!by.getName().equals(BySelect.CSS_SELECTOR)) - throw new IllegalArgumentException("此css selector语法不受支持,请换成xpath"); - String loc = "xpath:./ancestor::" + by.getValue().replaceAll("^[.\\s/]+", "") + "[" + index + "]"; - return _ele(loc, null, 1, true, false, "parent()").get(0); - } - - /** - * 返回上面某一级父元素,指定层数 - * - * @param level 第几级父元素 - * @return 上级元素对象 - */ - @Override - public T parent(Integer level) { - String loc = "xpath:./ancestor::*[" + level + "]"; - return _ele(loc, null, 1, true, false, "parent()").get(0); - } - - /** - * 返回上面某一级父元素 用查询语法定位 - * - * @param loc 定位符 - * @param index 选择第几个结果 - * @return 上级元素对象 - */ - @Override - public T parent(String loc, Integer index) { - By loc1 = Locator.getLoc(loc, true, false); - if (!loc1.getName().equals(BySelect.CSS_SELECTOR)) - throw new IllegalArgumentException("此css selector语法不受支持,请换成xpath"); - loc = "xpath:./ancestor::" + loc1.getValue().replaceAll("^[.\\s/]+", "") + "[" + index + "]"; - return _ele(loc, null, 1, true, false, "parent()").get(0); - } - - /** - * 返回直接子元素元素或节点组成的列表,可用查询语法筛选 - * - * @param index 第几个查询结果,1开始 - * @return 直接子元素或节点文本组成的列表 - */ - public T child(Integer index) { - return child(index, null); - } - - /** - * 返回直接子元素元素或节点组成的列表,可用查询语法筛选 - * - * @param index 第几个查询结果,1开始 - * @param timeout 查找节点的超时时间 - * @return 直接子元素或节点文本组成的列表 - */ - public T child(Integer index, Double timeout) { - return child(index, timeout, true); - } - - /** - * 返回直接子元素元素或节点组成的列表,可用查询语法筛选 - * - * @param index 第几个查询结果,1开始 - * @param timeout 查找节点的超时时间 - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 直接子元素或节点文本组成的列表 - */ - public T child(Integer index, Double timeout, Boolean eleOnly) { - String loc = eleOnly ? "*" : "node()"; - List ts = this._ele("xpath./" + loc, timeout, index, true, false, null); - if (!ts.isEmpty()) { - return ts.get(0); - } else if (Settings.raiseWhenEleNotFound) { - throw new ElementNotFoundError("child()", Map.of("loc", "", "index", index, "eleOnly", eleOnly)); - } - return null; - } - - /** - * 返回直接子元素元素或节点组成的列表,可用查询语法筛选 - * - * @return 直接子元素或节点文本组成的列表 - */ - public T child() { - return child(""); - } - - /** - * 返回直接子元素元素或节点组成的列表,可用查询语法筛选 - * - * @param loc 查询语句 - * @return 直接子元素或节点文本组成的列表 - */ - public T child(String loc) { - return child(loc.isEmpty() ? null : loc, 1); - } - - /** - * 返回直接子元素元素或节点组成的列表,可用查询语法筛选 - * - * @param loc 查询语句 - * @param index 第几个查询结果,1开始 - * @return 直接子元素或节点文本组成的列表 - */ - public T child(String loc, Integer index) { - return child(loc, index, null); - } - - /** - * 返回直接子元素元素或节点组成的列表,可用查询语法筛选 - * - * @param loc 查询语句 - * @param index 第几个查询结果,1开始 - * @param timeout 查找节点的超时时间 - * @return 直接子元素或节点文本组成的列表 - */ - public T child(String loc, Integer index, Double timeout) { - return child(loc, index, timeout, true); - } - - /** - * 返回直接子元素元素或节点组成的列表,可用查询语法筛选 - * - * @param loc 查询语句 - * @param index 第几个查询结果,1开始 - * @param timeout 查找节点的超时时间 - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 直接子元素或节点文本组成的列表 - */ - public T child(String loc, Integer index, Double timeout, Boolean eleOnly) { - if (loc == null || loc.isEmpty()) { - loc = eleOnly ? "*" : "node()"; - } else { - By by = Locator.getLoc(loc, true, false); - if (by.getName().equals(BySelect.CSS_SELECTOR)) - throw new IllegalArgumentException("此css selector语法不受支持,请换成xpath。"); - else loc = by.getValue().replaceAll("^[.\\s/]+", ""); - } - - List ts = this._ele("xpath./" + loc, timeout, index, true, false, null); - if (!ts.isEmpty()) { - return ts.get(0); - } else if (Settings.raiseWhenEleNotFound) { - throw new ElementNotFoundError("child()", Map.of("loc", loc, "index", index, "eleOnly", eleOnly)); - } - return null; - } - - /** - * 返回直接子元素元素或节点组成的列表,可用查询语法筛选 - * - * @param by 查询语句 - * @return 直接子元素或节点文本组成的列表 - */ - public T child(By by) { - return child(by, 1); - } - - /** - * 返回直接子元素元素或节点组成的列表,可用查询语法筛选 - * - * @param by 查询语句 - * @param index 第几个查询结果,1开始 - * @return 直接子元素或节点文本组成的列表 - */ - public T child(By by, Integer index) { - return child(by, index, null); - } - - /** - * 返回直接子元素元素或节点组成的列表,可用查询语法筛选 - * - * @param by 查询语句 - * @param index 第几个查询结果,1开始 - * @param timeout 查找节点的超时时间 - * @return 直接子元素或节点文本组成的列表 - */ - public T child(By by, Integer index, Double timeout) { - return child(by, index, timeout, true); - } - - /** - * 返回直接子元素元素或节点组成的列表,可用查询语法筛选 - * - * @param by 查询语句 - * @param index 第几个查询结果,1开始 - * @param timeout 查找节点的超时时间 - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 直接子元素或节点文本组成的列表 - */ - public T child(By by, Integer index, Double timeout, Boolean eleOnly) { - String loc; - if (by == null) { - loc = eleOnly ? "*" : "node()"; - } else { - by = Locator.getLoc(by, true, false); - if (by.getName().equals(BySelect.CSS_SELECTOR)) - throw new IllegalArgumentException("此css selector语法不受支持,请换成xpath。"); - else loc = by.getValue().replaceAll("^[.\\s/]+", ""); - } - - List ts = this._ele("xpath./" + loc, timeout, index, true, false, null); - if (!ts.isEmpty()) { - return ts.get(0); - } else if (Settings.raiseWhenEleNotFound) { - throw new ElementNotFoundError("child()", Map.of("loc", loc, "index", index, "eleOnly", eleOnly)); - } - return null; - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @return 兄弟元素 - */ - public T prev() { - return prev(""); - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param loc 用于筛选的查询语法 - * @return 兄弟元素 - */ - public T prev(String loc) { - return prev(loc.isEmpty() ? null : loc, null); - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param loc 用于筛选的查询语法 - * @param index 前面第几个查询结果,0开始 - * @return 兄弟元素 - */ - - public T prev(String loc, Integer index) { - return prev(loc, index, null); - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param loc 用于筛选的查询语法 - * @param index 前面第几个查询结果,0开始 - * @param timeout 查找节点的超时时间(秒) - * @return 兄弟元素 - */ - public T prev(String loc, Integer index, Double timeout) { - return prev(loc, index, timeout, true); - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param loc 用于筛选的查询语法 - * @param index 前面第几个查询结果,1开始 - * @param timeout 查找节点的超时时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 兄弟元素 - */ - public T prev(String loc, Integer index, Double timeout, Boolean eleOnly) { - return this.getRelative("prev()", "preceding", true, loc, index, timeout, eleOnly); - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param by 用于筛选的元素 - * @return 兄弟元素 - */ - public T prev(By by) { - return prev(by, 1); - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param by 用于筛选的元素 - * @param index 前面第几个查询结果,0开始 - * @return 兄弟元素 - */ - public T prev(By by, Integer index) { - return prev(by, index, null); - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param by 用于筛选的元素 - * @param index 前面第几个查询结果,0开始 - * @param timeout 查找节点的超时时间(秒) - * @return 兄弟元素 - */ - public T prev(By by, Integer index, Double timeout) { - return prev(by, index, timeout, true); - } - - public T prev(By by, Integer index, Double timeout, Boolean eleOnly) { - return this.getRelative("prev()", "preceding", true, by, index, timeout, eleOnly); - } - - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param index 前面第几个查询结果,0开始 - * @return 兄弟元素 - */ - public T prev(Integer index) { - return prev(index, null); - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param index 前面第几个查询结果,0开始 - * @param timeout 查找节点的超时时间(秒) - * @return 兄弟元素 - */ - public T prev(Integer index, Double timeout) { - return prev(index, timeout, true); - } - - public T prev(Integer index, Double timeout, Boolean eleOnly) { - return this.getRelative("prev()", "preceding", true, index, timeout, eleOnly); - } - - /** - * 返回后面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param by 用于筛选的查询语法 - * @param index 第几个查询结果,1开始 - * @param timeout 查找节点的超时时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 兄弟元素 - */ - @Override - public T next(By by, Integer index, Double timeout, Boolean eleOnly) { - return this.getRelative("next()", "following", true, by, index, timeout, eleOnly); - } - - /** - * 返回后面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param loc 用于筛选的查询语法 - * @param index 第几个查询结果,1开始 - * @param timeout 查找节点的超时时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 兄弟元素 - */ - - @Override - public T next(String loc, Integer index, Double timeout, Boolean eleOnly) { - return this.getRelative("next()", "following", true, loc, index, timeout, eleOnly); - } - - /** - * 返回当前元素后面一个符合条件的同级元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param index 第几个查询结果,0开始 - * @param timeout 查找节点的超时时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 兄弟元素或节点文本 - */ - @Override - public T next(Integer index, Double timeout, Boolean eleOnly) { - return this.getRelative("next()", "following", true, index, timeout, eleOnly); - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param by 用于筛选的查询语法 - * @return 本元素前面的某个元素或节点 - */ - public T before(By by) { - return before(by, null); - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param by 用于筛选的查询语法 - * @param index 前面第几个查询结果,1开始 - * @return 本元素前面的某个元素或节点 - */ - public T before(By by, Integer index) { - return before(by, index, null); - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param by 用于筛选的查询语法 - * @param index 前面第几个查询结果,1开始 - * @param timeout 查找节点的超时时间(秒) - * @return 本元素前面的某个元素或节点 - */ - public T before(By by, Integer index, Double timeout) { - return before(by, index, timeout, true); - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param by 用于筛选的查询语法 - * @param index 前面第几个查询结果,1开始 - * @param timeout 查找节点的超时时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 本元素前面的某个元素或节点 - */ - public T before(By by, Integer index, Double timeout, Boolean eleOnly) { - return this.getRelative("before()", "preceding", false, by, index, timeout, eleOnly); - } - - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @return 本元素前面的某个元素或节点 - */ - public T before() { - return before(""); - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param loc 用于筛选的查询语法 - * @return 本元素前面的某个元素或节点 - */ - public T before(String loc) { - return before(loc.isEmpty() ? null : loc, null); - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param loc 用于筛选的查询语法 - * @param index 前面第几个查询结果,1开始 - * @return 本元素前面的某个元素或节点 - */ - public T before(String loc, Integer index) { - return before(loc, index, null); - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param loc 用于筛选的查询语法 - * @param index 前面第几个查询结果,1开始 - * @param timeout 查找节点的超时时间(秒) - * @return 本元素前面的某个元素或节点 - */ - public T before(String loc, Integer index, Double timeout) { - return before(loc, index, timeout, true); - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param loc 用于筛选的查询语法 - * @param index 前面第几个查询结果,1开始 - * @param timeout 查找节点的超时时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 本元素前面的某个元素或节点 - */ - public T before(String loc, Integer index, Double timeout, Boolean eleOnly) { - return this.getRelative("before()", "preceding", false, loc, index, timeout, eleOnly); - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param index 前面第几个查询结果,1开始 - * @return 本元素前面的某个元素或节点 - */ - public T before(Integer index) { - return before(index, null); - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param index 前面第几个查询结果,1开始 - * @param timeout 查找节点的超时时间(秒) - * @return 本元素前面的某个元素或节点 - */ - public T before(Integer index, Double timeout) { - return before(index, timeout, true); - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param index 前面第几个查询结果,1开始 - * @param timeout 查找节点的超时时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 本元素前面的某个元素或节点 - */ - public T before(Integer index, Double timeout, Boolean eleOnly) { - return this.getRelative("before()", "preceding", false, index, timeout, eleOnly); - } - - /** - * 返回后面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param by 用于筛选的查询语法 - * @return 本元素后面的某个元素或节点 - */ - public T after(By by) { - return after(by, null); - } - - /** - * 返回后面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param by 用于筛选的查询语法 - * @param index 前面第几个查询结果,1开始 - * @return 本元素后面的某个元素或节点 - */ - public T after(By by, Integer index) { - return after(by, index, null); - } - - /** - * 返回后面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param by 用于筛选的查询语法 - * @param index 前面第几个查询结果,1开始 - * @param timeout 查找节点的超时时间(秒) - * @return 本元素后面的某个元素或节点 - */ - public T after(By by, Integer index, Double timeout) { - return after(by, index, timeout, true); - } - - /** - * 返回后面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param by 用于筛选的查询语法 - * @param index 前面第几个查询结果,1开始 - * @param timeout 查找节点的超时时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 本元素后面的某个元素或节点 - */ - public T after(By by, Integer index, Double timeout, Boolean eleOnly) { - return this.getRelative("after()", "following", false, by, index, timeout, eleOnly); - } - - /** - * 返回后面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @return 本元素后面的某个元素或节点 - */ - public T after() { - return after(""); - } - - /** - * 返回后面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param loc 用于筛选的查询语法 - * @return 本元素后面的某个元素或节点 - */ - public T after(String loc) { - return after(loc.isEmpty() ? null : loc, null); - } - - /** - * 返回后面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param loc 用于筛选的查询语法 - * @param index 前面第几个查询结果,1开始 - * @return 本元素后面的某个元素或节点 - */ - public T after(String loc, Integer index) { - return after(loc, index, null); - } - - /** - * 返回后面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param loc 用于筛选的查询语法 - * @param index 前面第几个查询结果,1开始 - * @param timeout 查找节点的超时时间(秒) - * @return 本元素后面的某个元素或节点 - */ - public T after(String loc, Integer index, Double timeout) { - return after(loc, index, timeout, true); - } - - /** - * 返回后面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param loc 用于筛选的查询语法 - * @param index 前面第几个查询结果,1开始 - * @param timeout 查找节点的超时时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 本元素后面的某个元素或节点 - */ - public T after(String loc, Integer index, Double timeout, Boolean eleOnly) { - return this.getRelative("after()", "following", false, loc, index, timeout, eleOnly); - } - - /** - * 返回后面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param index 前面第几个查询结果,1开始 - * @return 本元素后面的某个元素或节点 - */ - public T after(Integer index) { - return after(index, null); - } - - /** - * 返回后面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param index 前面第几个查询结果,1开始 - * @param timeout 查找节点的超时时间(秒) - * @return 本元素后面的某个元素或节点 - */ - public T after(Integer index, Double timeout) { - return after(index, timeout, true); - } - - /** - * 返回后面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param index 前面第几个查询结果,1开始 - * @param timeout 查找节点的超时时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 本元素后面的某个元素或节点 - */ - public T after(Integer index, Double timeout, Boolean eleOnly) { - return this.getRelative("after()", "following", false, index, timeout, eleOnly); - } - - /** - * 返回直接子元素元素或节点组成的列表,可用查询语法筛选 - * - * @param by 用于筛选的查询语法 - * @return 本元素后面的某个元素或节点 - */ - public List children(By by) { - return children(by, null); - } - - /** - * 返回直接子元素元素或节点组成的列表,可用查询语法筛选 - * - * @param by 用于筛选的查询语法 - * @param timeout 查找节点的超时时间(秒) - * @return 本元素后面的某个元素或节点 - */ - public List children(By by, Double timeout) { - return children(by, timeout, true); - } - - /** - * 返回直接子元素元素或节点组成的列表,可用查询语法筛选 - * - * @param by 用于筛选的查询语法 - * @param timeout 查找节点的超时时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 本元素后面的某个元素或节点 - */ - public List children(By by, Double timeout, Boolean eleOnly) { - String loc; - if (by == null) { - loc = eleOnly ? "*" : "node()"; - } else { - by = Locator.getLoc(by, true, false); - if (by.getName().equals(BySelect.CSS_SELECTOR)) - throw new IllegalArgumentException("此css selector语法不受支持,请换成xpath。"); - else loc = by.getValue().replaceAll("^[.\\s/]+", ""); - } - loc = "xpath:./" + loc; - return this._ele(loc, timeout, null, true, null, null); - } - - /** - * 返回直接子元素元素或节点组成的列表,可用查询语法筛选 - * - * @return 本元素后面的某个元素或节点 - */ - public List children() { - return children(""); - } - - /** - * 返回直接子元素元素或节点组成的列表,可用查询语法筛选 - * - * @param loc 用于筛选的查询语法 - * @return 本元素后面的某个元素或节点 - */ - public List children(String loc) { - return children(loc.isEmpty() ? null : loc, null); - } - - /** - * 返回直接子元素元素或节点组成的列表,可用查询语法筛选 - * - * @param loc 用于筛选的查询语法 - * @param timeout 查找节点的超时时间(秒) - * @return 本元素后面的某个元素或节点 - */ - public List children(String loc, Double timeout) { - return children(loc, timeout, true); - } - - /** - * 返回直接子元素元素或节点组成的列表,可用查询语法筛选 - * - * @param loc 用于筛选的查询语法 - * @param timeout 查找节点的超时时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 本元素后面的某个元素或节点 - */ - public List children(String loc, Double timeout, Boolean eleOnly) { - if (loc == null || loc.isEmpty()) { - loc = eleOnly ? "*" : "node()"; - } else { - By by = Locator.getLoc(loc, true, false); - if (by.getName().equals(BySelect.CSS_SELECTOR)) - throw new IllegalArgumentException("此css selector语法不受支持,请换成xpath。"); - else { - loc = by.getValue().replaceAll("^[.\\s/]+", ""); - } - } - loc = "xpath:./" + loc; - return this._ele(loc, timeout, null, true, null, null); - } - - - /** - * 获取前面符合条件的同级列表 - * - * @return 同级元素或节点文本组成的列表 - */ - public List prevs() { - return prevs("", null); - } - - /** - * 获取前面符合条件的同级列表 - * - * @param loc 查询元素 - * @return 同级元素或节点文本组成的列表 - */ - public List prevs(String loc) { - return prevs(loc.isEmpty() ? null : loc, null); - } - - /** - * 获取前面符合条件的同级列表 - * - * @param loc 查询元素 - * @param timeout 等待时间 - * @return 同级元素或节点文本组成的列表 - */ - public List prevs(String loc, Double timeout) { - return prevs(loc, timeout, true); - } - - /** - * 回前面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param loc 查询元素 - * @param timeout 等待时间 - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 兄弟元素的列表 - */ - public List prevs(String loc, Double timeout, Boolean eleOnly) { - return this.getRelatives(null, loc, "preceding", true, timeout, eleOnly); - } - - /** - * 获取前面符合条件的同级列表 - * - * @param by 查询元素 - * @return 同级元素或节点文本组成的列表 - */ - public List prevs(By by) { - return prevs(by, null); - } - - /** - * 获取前面符合条件的同级列表 - * - * @param by 查询元素 - * @param timeout 等待时间 - * @return 同级元素或节点文本组成的列表 - */ - public List prevs(By by, Double timeout) { - return prevs(by, timeout, true); - } - - - /** - * 回前面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param by 查询元素 - * @param timeout 等待时间 - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 兄弟元素的列表 - */ - public List prevs(By by, Double timeout, Boolean eleOnly) { - return this.getRelatives(null, by, "preceding", true, timeout, eleOnly); - } - - /** - * 返回后面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param by 用于筛选的查询语法 - * @return 兄弟元素或节点文本组成的列表 - */ - public List nexts(By by) { - return nexts(by, null); - } - - /** - * 返回后面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param by 用于筛选的查询语法 - * @param timeout 查找节点的超时时间 - * @return 兄弟元素或节点文本组成的列表 - */ - public List nexts(By by, Double timeout) { - return nexts(by, timeout, true); - } - - /** - * 返回后面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param by 用于筛选的查询语法 - * @param timeout 查找节点的超时时间 - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 兄弟元素或节点文本组成的列表 - */ - public List nexts(By by, Double timeout, Boolean eleOnly) { - return this.getRelatives(null, by, "following", true, timeout, eleOnly); - } - - /** - * 返回后面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @return 兄弟元素或节点文本组成的列表 - */ - - public List nexts() { - return nexts(""); - } - - /** - * 返回后面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param loc 用于筛选的查询语法 - * @return 兄弟元素或节点文本组成的列表 - */ - public List nexts(String loc) { - return nexts(loc.isEmpty() ? null : loc, null); - } - - /** - * 返回后面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param loc 用于筛选的查询语法 - * @param timeout 查找节点的超时时间 - * @return 兄弟元素或节点文本组成的列表 - */ - public List nexts(String loc, Double timeout) { - return nexts(loc, timeout, true); - } - - /** - * 返回后面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param loc 用于筛选的查询语法 - * @param timeout 查找节点的超时时间 - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 兄弟元素或节点文本组成的列表 - */ - public List nexts(String loc, Double timeout, Boolean eleOnly) { - return this.getRelatives(null, loc, "following", true, timeout, eleOnly); - } - - /** - * 返回后面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param by 用于筛选的查询语法 - * @return 本元素前面的元素或节点组成的列表 - */ - public List befores(By by) { - return this.befores(by, null); - } - - /** - * 返回后面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param by 用于筛选的查询语法 - * @param timeout 查找节点的超时时间(秒) - * @return 本元素前面的元素或节点组成的列表 - */ - public List befores(By by, Double timeout) { - return this.befores(by, timeout, true); - } - - /** - * 返回后面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param by 用于筛选的查询语法 - * @param timeout 查找节点的超时时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 本元素前面的元素或节点组成的列表 - */ - public List befores(By by, Double timeout, Boolean eleOnly) { - return this.getRelatives(null, by, "preceding", false, timeout, eleOnly); - } - - /** - * 返回后面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @return 本元素前面的元素或节点组成的列表 - */ - public List befores() { - return this.befores(""); - } - - /** - * 返回后面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param loc 用于筛选的查询语法 - * @return 本元素前面的元素或节点组成的列表 - */ - public List befores(String loc) { - return this.befores(loc.isEmpty() ? null : loc, null); - } - - /** - * 返回后面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param loc 用于筛选的查询语法 - * @param timeout 查找节点的超时时间(秒) - * @return 本元素前面的元素或节点组成的列表 - */ - public List befores(String loc, Double timeout) { - return this.befores(loc, timeout, true); - } - - /** - * 返回后面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param loc 用于筛选的查询语法 - * @param timeout 查找节点的超时时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 本元素前面的元素或节点组成的列表 - */ - public List befores(String loc, Double timeout, Boolean eleOnly) { - return this.getRelatives(null, loc, "preceding", false, timeout, eleOnly); - } - - /** - * 返回前面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param by 用于筛选的查询语法 - * @return 本元素前面的元素或节点组成的列表 - */ - public List afters(By by) { - return this.afters(by, null); - } - - - /** - * 返回前面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param by 用于筛选的查询语法 - * @param timeout 查找节点的超时时间(秒) - * @return 本元素前面的元素或节点组成的列表 - */ - public List afters(By by, Double timeout) { - return this.afters(by, timeout, true); - } - - /** - * 返回前面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param by 用于筛选的查询语法 - * @param timeout 查找节点的超时时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 本元素前面的元素或节点组成的列表 - */ - public List afters(By by, Double timeout, Boolean eleOnly) { - return this.getRelatives(null, by, "following", false, timeout, eleOnly); - } - - - /** - * 返回前面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @return 本元素前面的元素或节点组成的列表 - */ - public List afters() { - return this.afters(""); - } - - /** - * 返回前面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param loc 用于筛选的查询语法 - * @return 本元素前面的元素或节点组成的列表 - */ - public List afters(String loc) { - return this.afters(loc.isEmpty() ? null : loc, null); - } - - /** - * 返回前面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param loc 用于筛选的查询语法 - * @param timeout 查找节点的超时时间(秒) - * @return 本元素前面的元素或节点组成的列表 - */ - public List afters(String loc, Double timeout) { - return this.afters(loc, timeout, true); - } - - /** - * 返回前面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param loc 用于筛选的查询语法 - * @param timeout 查找节点的超时时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 本元素前面的元素或节点组成的列表 - */ - public List afters(String loc, Double timeout, Boolean eleOnly) { - return this.getRelatives(null, loc, "following", false, timeout, eleOnly); - } - - /** - * *获取一个亲戚元素或节点,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param func: 方法名称 - * @param direction: 方向,'following' 或 'preceding' - * @param brother 查找范围,在同级查找还是整个dom前后查找 - * @param loc: 用于筛选的查询语法 - * @param timeout: 查找节点的超时时间(秒) - * @param eleOnly: 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 本元素前面的某个元素或节点 - */ - private T getRelative(String func, String direction, Boolean brother, Integer loc, Double timeout, Boolean eleOnly) { - List relatives = this.getRelatives(loc, "", direction, brother, timeout, eleOnly); - if (!relatives.isEmpty()) return relatives.get(0); - if (Settings.raiseWhenEleNotFound) - throw new ElementNotFoundError(func, Map.of("loc", "", "index", loc, "eleOnly", eleOnly)); - return null; - } - - /** - * *获取一个亲戚元素或节点,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param func: 方法名称 - * @param direction: 方向,'following' 或 'preceding' - * @param brother 查找范围,在同级查找还是整个dom前后查找 - * @param loc: 用于筛选的查询语法 - * @param index: 前面第几个查询结果,1开始 - * @param timeout: 查找节点的超时时间(秒) - * @param eleOnly: 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 本元素前面的某个元素或节点 - */ - private T getRelative(String func, String direction, Boolean brother, String loc, Integer index, Double timeout, Boolean eleOnly) { - List relatives = this.getRelatives(index, loc, direction, brother, timeout, eleOnly); - if (!relatives.isEmpty()) return relatives.get(0); - if (Settings.raiseWhenEleNotFound) - throw new ElementNotFoundError(func, Map.of("loc", "", "index", index, "eleOnly", eleOnly)); - return null; - } - - /** - * *获取一个亲戚元素或节点,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param func: 方法名称 - * @param direction: 方向,'following' 或 'preceding' - * @param brother 查找范围,在同级查找还是整个dom前后查找 - * @param by: 用于筛选的查询语法 - * @param index: 前面第几个查询结果,1开始 - * @param timeout: 查找节点的超时时间(秒) - * @param eleOnly: 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 本元素前面的某个元素或节点 - */ - private T getRelative(String func, String direction, Boolean brother, By by, Integer index, Double timeout, Boolean eleOnly) { - List relatives = this.getRelatives(index, by, direction, brother, timeout, eleOnly); - if (!relatives.isEmpty()) return relatives.get(0); - if (Settings.raiseWhenEleNotFound) - throw new ElementNotFoundError(func, Map.of("loc", "", "index", index, "eleOnly", eleOnly)); - return null; - } - - /** - * @param index 获取第几个,该参数不为null时只获取该编号的元素 - * @param loc 用于筛选的查询语法 - * @param direction 'following' 或 'preceding',查找的方向 - * @param brother 查找范围,在同级查找还是整个dom前后查找 - * @param timeout 查找等待时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 元素对象或字符串 - */ - private List getRelatives(Integer index, String loc, String direction, Boolean brother, Double timeout, Boolean eleOnly) { - //获取查到范围 - String brotherStr = brother ? "-sibling" : ""; - if (loc == null || loc.isEmpty()) { - loc = eleOnly ? "*" : "node()"; - } else { - By by = Locator.getLoc(loc, true, false); - if (!by.getName().equals(BySelect.CSS_SELECTOR)) - throw new IllegalArgumentException("此css selector语法不受支持,请换成xpath。"); - else loc = by.getValue().replaceAll("^[.\\s/]+", ""); - } - loc = "xpath:./" + direction + brotherStr + "::" + loc; - if (index != null) index = "following".equals(direction) ? index : -index; - return this._ele(loc, timeout, index, true, false, null); - } - - /** - * @param index 获取第几个,该参数不为null时只获取该编号的元素 - * @param by 用于筛选的查询语法 - * @param direction 'following' 或 'preceding',查找的方向 - * @param brother 查找范围,在同级查找还是整个dom前后查找 - * @param timeout 查找等待时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 元素对象或字符串 - */ - private List getRelatives(Integer index, By by, String direction, Boolean brother, Double timeout, Boolean eleOnly) { - //获取查到范围 - String brotherStr = brother ? "-sibling" : ""; - String loc; - if (by == null) { - loc = eleOnly ? "*" : "node()"; - } else { - by = Locator.getLoc(by, true, false); - if (!by.getName().equals(BySelect.CSS_SELECTOR)) - throw new IllegalArgumentException("此css selector语法不受支持,请换成xpath。"); - else loc = by.getValue().replaceAll("^[.\\s/]+", ""); - } - loc = "xpath:./" + direction + brotherStr + "::" + loc; - if (index != null) index = "following".equals(direction) ? index : -index; - return this._ele(loc, timeout, index, true, false, null); - } - - //---------------------------------------------- - - /** - * @return 返回元素所有属性及值 - */ - public abstract Map attrs(); - - /** - * 返回处理的元素内文本 - */ - public abstract String text(); - - /** - * 返回未格式化处理的元素内文本 - */ - public abstract String rawText(); - - /** - * 返回attribute属性值 - * - * @param attr 属性名 - * @return 属性值文本,没有该属性返回null - */ - public abstract String attr(String attr); - - /** - * 获取css路径或xpath路径 - * - * @param mode 'css' 或 'xpath' - * @return css路径或xpath路径 - */ - protected abstract String getElePath(ElePathMode mode); -} diff --git a/java/src/main/java/com/ll/DrissonPage/base/Driver.java b/java/src/main/java/com/ll/DrissonPage/base/Driver.java deleted file mode 100644 index fc53df3..0000000 --- a/java/src/main/java/com/ll/DrissonPage/base/Driver.java +++ /dev/null @@ -1,450 +0,0 @@ -package com.ll.DrissonPage.base; - -import com.alibaba.fastjson.JSON; -import com.alibaba.fastjson.JSONObject; -import com.ll.DrissonPage.functions.Settings; -import lombok.Getter; -import lombok.Setter; -import okhttp3.*; -import okio.ByteString; -import org.jetbrains.annotations.NotNull; - -import java.util.ArrayList; -import java.util.HashMap; -import java.util.Map; -import java.util.concurrent.BlockingQueue; -import java.util.concurrent.ConcurrentHashMap; -import java.util.concurrent.LinkedBlockingQueue; -import java.util.concurrent.TimeUnit; -import java.util.concurrent.atomic.AtomicBoolean; -import java.util.concurrent.atomic.AtomicInteger; -import java.util.concurrent.atomic.AtomicReference; - -/** - * 驱动 okHTTP框架 - * - * @author 陆 - * @address click - * @original DrissionPage - */ - -public class Driver { - /** - * 标签id - */ - @Getter - private final String id; - /** - * 浏览器连接地址 - */ - @Getter - private final String address; - /** - * 标签页类型 - */ - @Getter - private final String type; - @Setter - @Getter - private boolean debug; - private final String websocketUrl; - private final AtomicInteger curId; - /** - * 会话返回值 - */ - private final AtomicReference webSocketMsg = new AtomicReference<>(null); - private final Thread recvThread; - private final Thread handleEventThread; - @Getter - private AtomicBoolean stopped; - private final BlockingQueue> eventQueue; - private final BlockingQueue> immediateEventQueue; - private final ConcurrentHashMap eventHandlers; - private final ConcurrentHashMap immediateEventHandlers; - private final ConcurrentHashMap>> methodResults; - /** - * 创建这个驱动的对象 - */ - @Getter - @Setter - private Occupant occupant; - private boolean alertFlag; - private static final OkHttpClient okHttpClient = new OkHttpClient().newBuilder() - .connectTimeout(5, TimeUnit.SECONDS) - .callTimeout(5, TimeUnit.SECONDS) - .readTimeout(5, TimeUnit.SECONDS) - .pingInterval(5, TimeUnit.SECONDS).build(); - /** - * 会话驱动 - */ - private WebSocket ws; - private Thread handleImmediateEventThread; - - public Driver(String tabId, String tabType, String address) { - this(tabId, tabType, address, null); - } - - /** - * 驱动 - * - * @param tabId 标签id - * @param tabType 标签页类型 - * @param address 浏览器连接地址 - */ - public Driver(String tabId, String tabType, String address, Occupant occupant) { - this.id = tabId; - this.address = address; - this.type = tabType; - this.occupant = occupant; - this.debug = false; - this.alertFlag = false; - this.websocketUrl = "ws://" + address + "/devtools/" + tabType + "/" + tabId; - this.curId = new AtomicInteger(0); - this.ws = null; - - - this.recvThread = new Thread(this::recvLoop); - this.handleEventThread = new Thread(this::handleEventLoop); - this.recvThread.setDaemon(true); - this.handleEventThread.setDaemon(true); - this.handleImmediateEventThread = null; - - this.stopped = new AtomicBoolean(); - - this.eventHandlers = new ConcurrentHashMap<>(); - this.immediateEventHandlers = new ConcurrentHashMap<>(); - this.methodResults = new ConcurrentHashMap<>(); - this.eventQueue = new LinkedBlockingQueue<>(); - this.immediateEventQueue = new LinkedBlockingQueue<>(); - start(); - } - - /** - * 发送信息到浏览器,并返回浏览器返回的信息 - * - * @param message 发送给浏览器的数据 - * @param timeout 超时时间,为null表示无时间 - * @return 浏览器返回的数据 - */ - private JSONObject send(Map message, double timeout) { - message = new HashMap<>(message); - int wsId = curId.incrementAndGet(); - message.put("id", wsId); - String messageJson = JSON.toJSONString(message); - - if (this.debug) System.out.println("发->" + messageJson); - //计算等待时间 - long endTime = (long) (System.currentTimeMillis() + timeout * 1000L); - LinkedBlockingQueue> value = new LinkedBlockingQueue<>(); - methodResults.put(wsId, value); - try { - ws.send(messageJson); - if (timeout == 0) { - methodResults.remove(wsId); - return new JSONObject(Map.of("id", wsId, "result", Map.of())); - } - } catch (Exception e) { - e.printStackTrace(); - methodResults.remove(wsId); - return new JSONObject(Map.of("error", Map.of("message", "connection disconnected"), "type", "connection_error")); - } - int i = 5; - long endTimes = System.currentTimeMillis() + 1000L; - while (!stopped.get()) { - try { - Map result = methodResults.get(wsId).poll(5, TimeUnit.SECONDS); - if (result == null && System.currentTimeMillis() < endTimes) continue; - if (result == null && i > 0 && System.currentTimeMillis() > endTimes) { - i--; - endTimes = System.currentTimeMillis() + 1000L; - if (debug) System.out.println("超时或者丢包,重新发送:->" + messageJson); - ws.send(messageJson); - continue; - } - methodResults.remove(wsId); - if (result == null) throw new NullPointerException(); - return new JSONObject(result); - } catch (InterruptedException | NullPointerException | IllegalArgumentException e) { -// e.printStackTrace(); - String string = message.get("method").toString(); - if (alertFlag && string.startsWith("Input.") || string.startsWith("Runtime.")) { - return new JSONObject(Map.of("error", Map.of("message", "alert exists."), "type", "alert_exists")); - } - if (timeout > 0 && System.currentTimeMillis() > endTime) { - methodResults.remove(wsId); - return alertFlag ? new JSONObject(Map.of("error", Map.of("message", "alert exists."), "type", "alert_exists")) : new JSONObject(Map.of("error", Map.of("message", "timeout"), "type", "timeout")); - } - } - } - - return new JSONObject(Map.of("error", Map.of("message", "connection disconnected"), "type", "connection_error")); - } - - /** - * 接收浏览器信息的守护线程方法 - */ - private void recvLoop() { - while (!stopped.get()) { - JSONObject msg; - try { - String andSet = webSocketMsg.getAndSet(null); - if (andSet != null) { - msg = JSONObject.parseObject(andSet); - } else continue; - } catch (Exception e) { - if (stop()) return; - return; - - } - if (this.debug) System.out.println("<-收" + msg); - - if (msg.containsKey("method")) { - if (msg.getString("method").startsWith("Page.javascriptDialog")) { - alertFlag = msg.getString("method").endsWith("Opening"); - } - MyRunnable function = immediateEventHandlers.get(msg.getString("method")); - if (function != null) { - this.handleImmediateEvent(function, msg.getOrDefault("params", new HashMap<>())); - } else { - try { - eventQueue.put(msg); - } catch (InterruptedException e) { - throw new RuntimeException(e); - } - } - } else { - int i = 1000; - Integer integer = msg.getInteger("id"); - while (i-- > 0 && integer != null && !methodResults.containsKey(integer)) { - try { - Thread.sleep(10); - } catch (InterruptedException e) { - throw new RuntimeException(e); - } - } - if (methodResults.containsKey(integer)) { - try { - methodResults.get(integer).put(msg); - } catch (InterruptedException e) { - throw new RuntimeException(e); - } - } else if (this.debug) { - System.out.println("未知错误->" + msg); - - } - } - - } - } - - /** - * 当接收到浏览器信息,执行已绑定的方法 - */ - private void handleEventLoop() { - while (!stopped.get()) { - Map event; - try { - event = eventQueue.poll(2, TimeUnit.SECONDS); - } catch (InterruptedException e) { - continue; - } - - if (event != null) { - MyRunnable function = eventHandlers.get(event.get("method").toString()); - if (function != null) { - function.setMessage(event.get("params")); - function.run(); - } - } - this.eventQueue.poll(); - - } - } - - private void handleImmediateEventLoop() { - while (!stopped.get() && !immediateEventQueue.isEmpty()) { - Map event; - try { - event = immediateEventQueue.poll(2, TimeUnit.SECONDS); - } catch (InterruptedException e) { - e.printStackTrace(); - continue; - } - if (event != null) { - MyRunnable function = immediateEventHandlers.get(event.get("method").toString()); - if (function != null) { - function.setMessage(event.get("params")); - function.run(); - } - } - - } - } - - /** - * 处理立即执行的动作 - * - * @param function 要运行下方法 - * @param params 方法参数 - */ - private void handleImmediateEvent(MyRunnable function, Object params) { - Map func = new HashMap<>(); - func.put("method", function); - func.put("params", params); - immediateEventQueue.add(func); - - if (handleImmediateEventThread == null || !handleImmediateEventThread.isAlive()) { - handleImmediateEventThread = new Thread(this::handleImmediateEventLoop); - handleImmediateEventThread.setDaemon(true); - handleImmediateEventThread.start(); - } - } - - /** - * 执行cdp方法 - * - * @param method 方法 - * @return 执行结果 - */ - public Object run(String method) { - return run(method, new HashMap<>()); - } - - /** - * 执行cdp方法 - * - * @param method 方法 - * @param params 参数 - * @return 执行结果 - */ - public Object run(String method, Map params) { - if (stopped.get()) return Map.of("error", "connection disconnected", "type", "connection_error"); - params = new HashMap<>(params); - Object timeout1 = params.remove("_timeout"); - double timeout = timeout1 != null ? Float.parseFloat(timeout1.toString()) : Settings.cdpTimeout; - - JSONObject result = this.send(Map.of("method", method, "params", params), timeout); - if (!result.containsKey("result") && result.containsKey("error")) { - HashMap map = new HashMap<>(); - map.put("error", result.getJSONObject("error").get("message")); - map.put("type", result.getOrDefault("type", "call_method_error")); - map.put("method", method); - map.put("args", params); - map.put("timeout", timeout); - return JSON.toJSONString(map); - } else { - return JSON.toJSONString(result.get("result")); - } - } - - - /** - * 启动连接 - */ - private void start() { - this.stopped.set(false); - try { - Request build = new Request(new HttpUrl("socket", "", "", "", 80, new ArrayList<>(), null, null, this.websocketUrl), "GET", Headers.of(), null, new HashMap<>()).newBuilder().url(this.websocketUrl).build(); - ws = okHttpClient.newWebSocket(build, new WebSocketListener() { - @Override - public void onClosed(@NotNull WebSocket webSocket, int code, @NotNull String reason) { - // 关闭事件处理 - stop(); - super.onClosed(webSocket, code, reason); - } - - @Override - public void onClosing(@NotNull WebSocket webSocket, int code, @NotNull String reason) { - super.onClosing(webSocket, code, reason); - } - - @Override - public void onFailure(@NotNull WebSocket webSocket, @NotNull Throwable t, Response response) { - super.onFailure(webSocket, t, response); - } - - @Override - public void onMessage(@NotNull WebSocket webSocket, @NotNull String text) { - webSocketMsg.set(text); - super.onMessage(webSocket, text); - } - - @Override - public void onMessage(@NotNull WebSocket webSocket, @NotNull ByteString bytes) { - super.onMessage(webSocket, bytes); - } - - @Override - public void onOpen(@NotNull WebSocket webSocket, @NotNull Response response) { - super.onOpen(webSocket, response); - } - - }); - recvThread.start(); - handleEventThread.start(); - } catch (Exception e) { - e.printStackTrace(); - stop(); - } - } - - /** - * 中断连接 - */ - public boolean stop() { - stop1(); - while (this.recvThread.isAlive() || this.handleEventThread.isAlive()) { - try { - Thread.sleep(100); - } catch (InterruptedException e) { - throw new RuntimeException(e); - } - } - return true; - } - - /** - * 中断连接 - */ - private void stop1() { - if (debug) System.out.println("关闭"); - if (stopped.get()) return; - stopped.set(!stopped.get()); - if (ws != null) { - ws.close(1000, ""); - ws = null; - } - try { - while (!eventQueue.isEmpty()) { - Map event = eventQueue.poll(); - MyRunnable method = eventHandlers.get(event.get("method").toString()); - if (method != null) { - method.setMessage(event.get("params")); - method.run(); - } - } - } catch (Exception ignored) { - } - eventHandlers.clear(); - methodResults.clear(); - eventQueue.clear(); - if (occupant != null) occupant.onDisconnect(); - } - - public void setCallback(String event, MyRunnable callback) { - setCallback(event, callback, false); - } - - /** - * 绑定cdp event和回调方法 - * - * @param event 方法名称 - * @param callback 绑定到cdp event的回调方法 - * @param immediate 是否要立即处理的动作 - */ - public void setCallback(String event, MyRunnable callback, boolean immediate) { - Map handler = immediate ? immediateEventHandlers : eventHandlers; - if (callback != null) handler.put(event, callback); - else handler.remove(event); - } -} \ No newline at end of file diff --git a/java/src/main/java/com/ll/DrissonPage/base/Driver2.java b/java/src/main/java/com/ll/DrissonPage/base/Driver2.java deleted file mode 100644 index 8d2230e..0000000 --- a/java/src/main/java/com/ll/DrissonPage/base/Driver2.java +++ /dev/null @@ -1,423 +0,0 @@ -package com.ll.DrissonPage.base; - -/** - * 驱动 - * - * @author 陆 - * @address click - * @original DrissionPage - */ - -public class Driver2 { -// /** -// * 标签id -// */ -// @Getter -// private final String id; -// /** -// * 浏览器连接地址 -// */ -// @Getter -// private final String address; -// /** -// * 标签页类型 -// */ -// @Getter -// private final String type; -// private final boolean debug; -// private final String websocketUrl; -// private final AtomicInteger curId; -// /** -// * 会话返回值 -// */ -// private final AtomicReference webSocketMsg = new AtomicReference<>(null); -// private final Thread recvThread; -// private final Thread handleEventThread; -// @Getter -// -// private final AtomicBoolean stopped; -// private final BlockingQueue> eventQueue; -// private final BlockingQueue> immediateEventQueue; -// private final Map eventHandlers; -// private final Map immediateEventHandlers; -// private final Map>> methodResults; -// /** -// * 创建这个驱动的对象 -// */ -// @Getter -// @Setter -// private Occupant occupant; -// private boolean alertFlag; -// /** -// * 会话驱动 -// */ -// private WebSocket ws; -// private Thread handleImmediateEventThread; -// -// public Driver2(String tabId, String tabType, String address) { -// this(tabId, tabType, address, null); -// } -// -// /** -// * 驱动 -// * -// * @param tabId 标签id -// * @param tabType 标签页类型 -// * @param address 浏览器连接地址 -// */ -// public Driver2(String tabId, String tabType, String address, Occupant occupant) { -// this.id = tabId; -// this.address = address; -// this.type = tabType; -// this.occupant = occupant; -// this.debug = true; -// this.alertFlag = false; -// this.websocketUrl = "ws://" + address + "/devtools/" + tabType + "/" + tabId; -// this.curId = new AtomicInteger(0); -// this.ws = null; -// -// -// this.recvThread = new Thread(this::recvLoop); -// this.handleEventThread = new Thread(this::handleEventLoop); -// this.recvThread.setDaemon(true); -// this.handleEventThread.setDaemon(true); -// this.handleImmediateEventThread = null; -// -// this.stopped = new AtomicBoolean(); -// -// this.eventHandlers = new ConcurrentHashMap<>(); -// this.immediateEventHandlers = new ConcurrentHashMap<>(); -// this.methodResults = new ConcurrentHashMap<>(); -// this.eventQueue = new LinkedBlockingQueue<>(); -// this.immediateEventQueue = new LinkedBlockingQueue<>(); -// start(); -// } -// -// /** -// * 发送信息到浏览器,并返回浏览器返回的信息 -// * -// * @param message 发送给浏览器的数据 -// * @param timeout 超时时间,为null表示无时间 -// * @return 浏览器返回的数据 -// */ -// private JSONObject send(Map message, double timeout) { -// message = new HashMap<>(message); -// int wsId = curId.incrementAndGet(); -// message.put("id", wsId); -// String messageJson = JSON.toJSONString(message); -// -// if (this.debug) System.out.println("发->" + messageJson); -// //计算等待时间 -// long endTime = (long) (System.currentTimeMillis() + timeout * 1000L); -// LinkedBlockingQueue> value = new LinkedBlockingQueue<>(); -// methodResults.put(wsId, value); -// try { -// ws.send(messageJson); -// if (timeout == 0) { -// methodResults.remove(wsId); -// return new JSONObject(Map.of("id", wsId, "result", Map.of())); -// } -// } catch (WebsocketNotConnectedException e) { -// e.printStackTrace(); -// methodResults.remove(wsId); -// return new JSONObject(Map.of("error", Map.of("message", "connection disconnected"), "type", "connection_error")); -// } -// int i = 5; -// long endTimes = System.currentTimeMillis() + 1000L; -// while (!stopped.get()) { -// try { -// Map result = methodResults.get(wsId).poll(200, TimeUnit.MILLISECONDS); -// if (result == null && System.currentTimeMillis() < endTimes) continue; -// if (result == null && i > 0 && System.currentTimeMillis() > endTimes) { -// i--; -// endTimes = System.currentTimeMillis() + 1000L; -// ws.send(messageJson); -// continue; -// } -// methodResults.remove(wsId); -// if (result == null) throw new NullPointerException(); -// return new JSONObject(result); -// } catch (InterruptedException | NullPointerException | IllegalArgumentException e) { -//// e.printStackTrace(); -// String string = message.get("method").toString(); -// if (alertFlag && string.startsWith("Input.") || string.startsWith("Runtime.")) { -// return new JSONObject(Map.of("error", Map.of("message", "alert exists."), "type", "alert_exists")); -// } -// if (timeout > 0 && System.currentTimeMillis() > endTime) { -// methodResults.remove(wsId); -// return alertFlag ? new JSONObject(Map.of("error", Map.of("message", "alert exists."), "type", "alert_exists")) : new JSONObject(Map.of("error", Map.of("message", "timeout"), "type", "timeout")); -// } -// } -// } -// -// return new JSONObject(Map.of("error", Map.of("message", "connection disconnected"), "type", "connection_error")); -// } -// -// /** -// * 接收浏览器信息的守护线程方法 -// */ -// private void recvLoop() { -// while (!stopped.get()) { -// JSONObject msg; -// try { -// String andSet = webSocketMsg.getAndSet(null); -// if (andSet != null) { -// msg = JSONObject.parseObject(andSet); -// } else continue; -// } catch (Exception e) { -// if (stop()) return; -// return; -// -// } -// if (this.debug) System.out.println("<-收" + msg); -// -// if (msg.containsKey("method")) { -// if (msg.getString("method").startsWith("Page.javascriptDialog")) { -// alertFlag = msg.getString("method").endsWith("Opening"); -// } -// MyRunnable function = immediateEventHandlers.get(msg.getString("method")); -// if (function != null) { -// this.handleImmediateEvent(function, msg.getOrDefault("params", new HashMap<>())); -// } else { -// try { -// eventQueue.put(msg); -// } catch (InterruptedException e) { -// throw new RuntimeException(e); -// } -// } -// } else { -// int i = 1000; -// Integer integer = msg.getInteger("id"); -// while (i-- > 0 && integer != null && !methodResults.containsKey(integer)) { -// try { -// Thread.sleep(10); -// } catch (InterruptedException e) { -// throw new RuntimeException(e); -// } -// } -// if (methodResults.containsKey(integer)) { -// try { -// methodResults.get(integer).put(msg); -// } catch (InterruptedException e) { -// throw new RuntimeException(e); -// } -// } else if (this.debug) { -// System.out.println("未知错误->" + msg); -// -// } -// } -// -// } -// } -// -// /** -// * 当接收到浏览器信息,执行已绑定的方法 -// */ -// private void handleEventLoop() { -// while (!stopped.get()) { -// Map event; -// try { -// event = eventQueue.poll(1, TimeUnit.SECONDS); -// } catch (InterruptedException e) { -// continue; -// } -// -// if (event != null) { -// MyRunnable function = eventHandlers.get(event.get("method").toString()); -// if (function != null) { -// function.setMessage(event.get("params")); -// function.run(); -// } -// } -// this.eventQueue.poll(); -// -// } -// } -// -// private void handleImmediateEventLoop() { -// while (!stopped.get() && !immediateEventQueue.isEmpty()) { -// Map event; -// try { -// event = immediateEventQueue.poll(1, TimeUnit.SECONDS); -// } catch (InterruptedException e) { -// e.printStackTrace(); -// continue; -// } -// if (event != null) { -// MyRunnable function = immediateEventHandlers.get(event.get("method").toString()); -// if (function != null) { -// function.setMessage(event.get("params")); -// function.run(); -// } -// } -// -// } -// } -// -// /** -// * 处理立即执行的动作 -// * -// * @param function 要运行下方法 -// * @param params 方法参数 -// */ -// private void handleImmediateEvent(MyRunnable function, Object params) { -// Map func = new HashMap<>(); -// func.put("method", function); -// func.put("params", params); -// immediateEventQueue.add(func); -// -// if (handleImmediateEventThread == null || !handleImmediateEventThread.isAlive()) { -// handleImmediateEventThread = new Thread(this::handleImmediateEventLoop); -// handleImmediateEventThread.setDaemon(true); -// handleImmediateEventThread.start(); -// } -// } -// -// /** -// * 执行cdp方法 -// * -// * @param method 方法 -// * @return 执行结果 -// */ -// public Object run(String method) { -// return run(method, new HashMap<>()); -// } -// -// /** -// * 执行cdp方法 -// * -// * @param method 方法 -// * @param params 参数 -// * @return 执行结果 -// */ -// public Object run(String method, Map params) { -// if (stopped.get()) return Map.of("error", "connection disconnected", "type", "connection_error"); -// params = new HashMap<>(params); -// Object timeout1 = params.remove("_timeout"); -// double timeout = timeout1 != null ? Float.parseFloat(timeout1.toString()) : 30.0; -// -// JSONObject result = this.send(Map.of("method", method, "params", params), timeout); -// if (!result.containsKey("result") && result.containsKey("error")) { -// HashMap map = new HashMap<>(); -// map.put("error", result.getJSONObject("error").get("message")); -// map.put("type", result.getOrDefault("type", "call_method_error")); -// map.put("method", method); -// map.put("args", params); -// map.put("timeout", timeout); -// return JSON.toJSONString(map); -// } else { -// return JSON.toJSONString(result.get("result")); -// } -// } -// -// -// /** -// * 启动连接 -// */ -// private void start() { -// this.stopped.set(false); -// try { -// Request build = new Request(new HttpUrl("socket","","","",80,new ArrayList<>(),null,null,this.websocketUrl), "GET", Headers.of(), null, new HashMap<>()).newBuilder().url(this.websocketUrl).build(); -// OkHttpClient okHttpClient = new OkHttpClient(); -// ws = okHttpClient.newWebSocket(build, new WebSocketListener() { -// @Override -// public void onClosed(@NotNull WebSocket webSocket, int code, @NotNull String reason) { -// // 关闭事件处理 -// stop(); -// super.onClosed(webSocket, code, reason); -// } -// -// @Override -// public void onClosing(@NotNull WebSocket webSocket, int code, @NotNull String reason) { -// super.onClosing(webSocket, code, reason); -// } -// -// @Override -// public void onFailure(@NotNull WebSocket webSocket, @NotNull Throwable t, @Nullable Response response) { -// super.onFailure(webSocket, t, response); -// } -// -// @Override -// public void onMessage(@NotNull WebSocket webSocket, @NotNull String text) { -// webSocketMsg.set(text); -// super.onMessage(webSocket, text); -// } -// -// @Override -// public void onMessage(@NotNull WebSocket webSocket, @NotNull ByteString bytes) { -// super.onMessage(webSocket, bytes); -// } -// -// @Override -// public void onOpen(@NotNull WebSocket webSocket, @NotNull Response response) { -// super.onOpen(webSocket, response); -// } -// -// }); -// recvThread.start(); -// handleEventThread.start(); -// } catch (Exception e) { -// e.printStackTrace(); -// stop(); -// } -// } -// -// /** -// * 中断连接 -// */ -// public boolean stop() { -// stop1(); -// while (this.recvThread.isAlive() || this.handleEventThread.isAlive()) { -// try { -// Thread.sleep(100); -// } catch (InterruptedException e) { -// throw new RuntimeException(e); -// } -// } -// return true; -// } -// -// /** -// * 中断连接 -// */ -// private void stop1() { -// if (stopped.get()) return; -// stopped.set(true); -// if (ws != null) { -// ws.close(1000, ""); -// ws = null; -// } -// try { -// while (!eventQueue.isEmpty()) { -// Map event = eventQueue.poll(); -// MyRunnable method = eventHandlers.get(event.get("method").toString()); -// if (method != null) { -// method.setMessage(event.get("params")); -// method.run(); -// } -// } -// } catch (Exception ignored) { -// } -// eventHandlers.clear(); -// methodResults.clear(); -// eventQueue.clear(); -// if (occupant != null) occupant.onDisconnect(); -// } -// -// public void setCallback(String event, MyRunnable callback) { -// setCallback(event, callback, false); -// } -// -// /** -// * 绑定cdp event和回调方法 -// * -// * @param event 方法名称 -// * @param callback 绑定到cdp event的回调方法 -// * @param immediate 是否要立即处理的动作 -// */ -// public void setCallback(String event, MyRunnable callback, boolean immediate) { -// Map handler = immediate ? immediateEventHandlers : eventHandlers; -// if (callback != null) handler.put(event, callback); -// else handler.remove(event); -// } -} \ No newline at end of file diff --git a/java/src/main/java/com/ll/DrissonPage/base/Driver3.java b/java/src/main/java/com/ll/DrissonPage/base/Driver3.java deleted file mode 100644 index e7b385c..0000000 --- a/java/src/main/java/com/ll/DrissonPage/base/Driver3.java +++ /dev/null @@ -1,426 +0,0 @@ -package com.ll.DrissonPage.base; - -/** - * 驱动 - * - * @author 陆 - * @address click - * @original DrissionPage - */ - -public class Driver3 { -// /** -// * 标签id -// */ -// @Getter -// private final String id; -// /** -// * 浏览器连接地址 -// */ -// @Getter -// private final String address; -// /** -// * 标签页类型 -// */ -// @Getter -// private final String type; -// private final boolean debug; -// private final String websocketUrl; -// private final AtomicInteger curId; -// /** -// * 会话返回值 -// */ -// private final AtomicReference webSocketMsg = new AtomicReference<>(null); -// private final Thread recvThread; -// private final Thread handleEventThread; -// @Getter -// -// private final AtomicBoolean stopped; -// private final BlockingQueue> eventQueue; -// private final BlockingQueue> immediateEventQueue; -// private final Map eventHandlers; -// private final Map immediateEventHandlers; -// private final Map>> methodResults; -// /** -// * 创建这个驱动的对象 -// */ -// @Getter -// @Setter -// private Occupant occupant; -// private boolean alertFlag; -// /** -// * 会话驱动 -// */ -// private WebSocketClient ws; -// private Future webSocketSession; -// private Thread handleImmediateEventThread; -// -// public Driver3(String tabId, String tabType, String address) { -// this(tabId, tabType, address, null); -// } -// -// /** -// * 驱动 -// * -// * @param tabId 标签id -// * @param tabType 标签页类型 -// * @param address 浏览器连接地址 -// */ -// public Driver3(String tabId, String tabType, String address, Occupant occupant) { -// this.id = tabId; -// this.address = address; -// this.type = tabType; -// this.occupant = occupant; -// this.debug = true; -// this.alertFlag = false; -// this.websocketUrl = "ws://" + address + "/devtools/" + tabType + "/" + tabId; -// this.curId = new AtomicInteger(0); -// this.ws = null; -// this.recvThread = new Thread(this::recvLoop); -// this.handleEventThread = new Thread(this::handleEventLoop); -// this.recvThread.setDaemon(true); -// this.handleEventThread.setDaemon(true); -// this.handleImmediateEventThread = null; -// -// this.stopped = new AtomicBoolean(); -// -// this.eventHandlers = new ConcurrentHashMap<>(); -// this.immediateEventHandlers = new ConcurrentHashMap<>(); -// this.methodResults = new ConcurrentHashMap<>(); -// this.eventQueue = new LinkedBlockingQueue<>(); -// this.immediateEventQueue = new LinkedBlockingQueue<>(); -// start(); -// } -// -// /** -// * 发送信息到浏览器,并返回浏览器返回的信息 -// * -// * @param message 发送给浏览器的数据 -// * @param timeout 超时时间,为null表示无时间 -// * @return 浏览器返回的数据 -// */ -// private JSONObject send(Map message, double timeout) { -// message = new HashMap<>(message); -// int wsId = curId.incrementAndGet(); -// message.put("id", wsId); -// String messageJson = JSON.toJSONString(message); -// -// if (this.debug) System.out.println("发->" + messageJson); -// //计算等待时间 -// long endTime = (long) (System.currentTimeMillis() + timeout * 1000L); -// LinkedBlockingQueue> value = new LinkedBlockingQueue<>(); -// methodResults.put(wsId, value); -// try { -// if (ws.isStopping()) return new JSONObject(); -// webSocketSession.get().getRemote().sendString(messageJson); -// if (timeout == 0) { -// methodResults.remove(wsId); -// return new JSONObject(Map.of("id", wsId, "result", Map.of())); -// } -// } catch (WebsocketNotConnectedException | IOException | InterruptedException | ExecutionException e) { -// e.printStackTrace(); -// methodResults.remove(wsId); -// return new JSONObject(Map.of("error", Map.of("message", "connection disconnected"), "type", "connection_error")); -// } -// int i = 5; -// long endTimes = System.currentTimeMillis() + 1000L; -// while (!stopped.get()) { -// try { -// Map result = methodResults.get(wsId).poll(2000, TimeUnit.MILLISECONDS); -// if (result == null && System.currentTimeMillis() < endTimes) continue; -// if (result == null && i > 0 && System.currentTimeMillis() > endTimes) { -// System.out.println("丢包:->" + messageJson); -// i--; -// endTimes = System.currentTimeMillis() + 1000L; -// if (ws.isStopping()) return new JSONObject(); -// webSocketSession.get().getRemote().sendString(messageJson); -// continue; -// } -// methodResults.remove(wsId); -// if (result == null) throw new NullPointerException(); -// return new JSONObject(result); -// } catch (InterruptedException | NullPointerException | IllegalArgumentException | IOException | -// ExecutionException e) { -//// e.printStackTrace(); -// String string = message.get("method").toString(); -// if (alertFlag && string.startsWith("Input.") || string.startsWith("Runtime.")) { -// return new JSONObject(Map.of("error", Map.of("message", "alert exists."), "type", "alert_exists")); -// } -// if (timeout > 0 && System.currentTimeMillis() > endTime) { -// methodResults.remove(wsId); -// return alertFlag ? new JSONObject(Map.of("error", Map.of("message", "alert exists."), "type", "alert_exists")) : new JSONObject(Map.of("error", Map.of("message", "timeout"), "type", "timeout")); -// } -// } -// } -// -// return new JSONObject(Map.of("error", Map.of("message", "connection disconnected"), "type", "connection_error")); -// } -// -// /** -// * 接收浏览器信息的守护线程方法 -// */ -// private void recvLoop() { -// while (!stopped.get()) { -// JSONObject msg; -// try { -// String andSet = webSocketMsg.getAndSet(null); -// if (andSet != null) { -// msg = JSONObject.parseObject(andSet); -// } else continue; -// } catch (Exception e) { -// if (stop()) return; -// return; -// -// } -// if (this.debug) System.out.println("<-收" + msg); -// -// if (msg.containsKey("method")) { -// if (msg.getString("method").startsWith("Page.javascriptDialog")) { -// alertFlag = msg.getString("method").endsWith("Opening"); -// } -// MyRunnable function = immediateEventHandlers.get(msg.getString("method")); -// if (function != null) { -// this.handleImmediateEvent(function, msg.getOrDefault("params", new HashMap<>())); -// } else { -// try { -// eventQueue.put(msg); -// } catch (InterruptedException e) { -// throw new RuntimeException(e); -// } -// } -// } else { -// int i = 1000; -// Integer integer = msg.getInteger("id"); -// while (i-- > 0 && integer != null && !methodResults.containsKey(integer)) { -// try { -// Thread.sleep(10); -// } catch (InterruptedException e) { -// throw new RuntimeException(e); -// } -// } -// if (methodResults.containsKey(integer)) { -// try { -// methodResults.get(integer).put(msg); -// } catch (InterruptedException e) { -// throw new RuntimeException(e); -// } -// } else if (this.debug) { -// System.out.println("未知错误->" + msg); -// -// } -// } -// -// } -// } -// -// /** -// * 当接收到浏览器信息,执行已绑定的方法 -// */ -// private void handleEventLoop() { -// while (!stopped.get()) { -// Map event; -// try { -// event = eventQueue.poll(1, TimeUnit.SECONDS); -// } catch (InterruptedException e) { -// continue; -// } -// -// if (event != null) { -// MyRunnable function = eventHandlers.get(event.get("method").toString()); -// if (function != null) { -// function.setMessage(event.get("params")); -// function.run(); -// } -// } -// this.eventQueue.poll(); -// -// } -// } -// -// private void handleImmediateEventLoop() { -// while (!stopped.get() && !immediateEventQueue.isEmpty()) { -// Map event; -// try { -// event = immediateEventQueue.poll(1, TimeUnit.SECONDS); -// } catch (InterruptedException e) { -// e.printStackTrace(); -// continue; -// } -// if (event != null) { -// MyRunnable function = immediateEventHandlers.get(event.get("method").toString()); -// if (function != null) { -// function.setMessage(event.get("params")); -// function.run(); -// } -// } -// -// } -// } -// -// /** -// * 处理立即执行的动作 -// * -// * @param function 要运行下方法 -// * @param params 方法参数 -// */ -// private void handleImmediateEvent(MyRunnable function, Object params) { -// Map func = new HashMap<>(); -// func.put("method", function); -// func.put("params", params); -// immediateEventQueue.add(func); -// -// if (handleImmediateEventThread == null || !handleImmediateEventThread.isAlive()) { -// handleImmediateEventThread = new Thread(this::handleImmediateEventLoop); -// handleImmediateEventThread.setDaemon(true); -// handleImmediateEventThread.start(); -// } -// } -// -// /** -// * 执行cdp方法 -// * -// * @param method 方法 -// * @return 执行结果 -// */ -// public Object run(String method) { -// return run(method, new HashMap<>()); -// } -// -// /** -// * 执行cdp方法 -// * -// * @param method 方法 -// * @param params 参数 -// * @return 执行结果 -// */ -// public Object run(String method, Map params) { -// if (stopped.get()) return Map.of("error", "connection disconnected", "type", "connection_error"); -// params = new HashMap<>(params); -// Object timeout1 = params.remove("_timeout"); -// double timeout = timeout1 != null ? Float.parseFloat(timeout1.toString()) : 30.0; -// -// JSONObject result = this.send(Map.of("method", method, "params", params), timeout); -// if (!result.containsKey("result") && result.containsKey("error")) { -// HashMap map = new HashMap<>(); -// map.put("error", result.getJSONObject("error").get("message")); -// map.put("type", result.getOrDefault("type", "call_method_error")); -// map.put("method", method); -// map.put("args", params); -// map.put("timeout", timeout); -// return JSON.toJSONString(map); -// } else { -// return JSON.toJSONString(result.get("result")); -// } -// } -// -// -// /** -// * 启动连接 -// */ -// private void start() { -// this.stopped.set(false); -// try { -//// Request build = new Request(new HttpUrl("socket","","","",80,new ArrayList<>(),null,null,this.websocketUrl), "GET", Headers.of(), null, new HashMap<>()).newBuilder().url(this.websocketUrl).build(); -//// OkHttpClient okHttpClient = new OkHttpClient(); -// ws = new WebSocketClient(); -// ws.setConnectTimeout(60_000); -// ws.start(); -// webSocketSession = ws.connect(new WebSocketAdapter() { -// @Override -// public void onWebSocketClose(int statusCode, String reason) { -// stop(); -// super.onWebSocketClose(statusCode, reason); -// } -// -// @Override -// public void onWebSocketConnect(Session sess) { -// super.onWebSocketConnect(sess); -// } -// -// @Override -// public void onWebSocketError(Throwable cause) { -// super.onWebSocketError(cause); -// } -// -// @Override -// public boolean isConnected() { -// return super.isConnected(); -// } -// -// @Override -// public void onWebSocketText(String message) { -// webSocketMsg.set(message); -// } -// -// }, URI.create(this.websocketUrl)); -// recvThread.start(); -// handleEventThread.start(); -// } catch (Exception e) { -// e.printStackTrace(); -// stop(); -// } -// } -// -// /** -// * 中断连接 -// */ -// public boolean stop() { -// stop1(); -// while (this.recvThread.isAlive() || this.handleEventThread.isAlive()) { -// try { -// Thread.sleep(100); -// } catch (InterruptedException e) { -// throw new RuntimeException(e); -// } -// } -// return true; -// } -// -// /** -// * 中断连接 -// */ -// private void stop1() { -// if (stopped.get()) return; -// stopped.set(true); -// if (ws != null) { -// try { -// ws.stop(); -// } catch (Exception e) { -// throw new RuntimeException(e); -// } -// ws = null; -// } -// try { -// while (!eventQueue.isEmpty()) { -// Map event = eventQueue.poll(); -// MyRunnable method = eventHandlers.get(event.get("method").toString()); -// if (method != null) { -// method.setMessage(event.get("params")); -// method.run(); -// } -// } -// } catch (Exception ignored) { -// } -// eventHandlers.clear(); -// methodResults.clear(); -// eventQueue.clear(); -// if (occupant != null) occupant.onDisconnect(); -// } -// -// public void setCallback(String event, MyRunnable callback) { -// setCallback(event, callback, false); -// } -// -// /** -// * 绑定cdp event和回调方法 -// * -// * @param event 方法名称 -// * @param callback 绑定到cdp event的回调方法 -// * @param immediate 是否要立即处理的动作 -// */ -// public void setCallback(String event, MyRunnable callback, boolean immediate) { -// Map handler = immediate ? immediateEventHandlers : eventHandlers; -// if (callback != null) handler.put(event, callback); -// else handler.remove(event); -// } -} \ No newline at end of file diff --git a/java/src/main/java/com/ll/DrissonPage/base/Driver_org_webSocket.java b/java/src/main/java/com/ll/DrissonPage/base/Driver_org_webSocket.java deleted file mode 100644 index 68780b2..0000000 --- a/java/src/main/java/com/ll/DrissonPage/base/Driver_org_webSocket.java +++ /dev/null @@ -1,424 +0,0 @@ -package com.ll.DrissonPage.base; - -/** - * 驱动 org.java-websocket - * - * @author 陆 - * @address click - * @original DrissionPage - */ - -public class Driver_org_webSocket { -// /** -// * 标签id -// */ -// @Getter -// private final String id; -// /** -// * 浏览器连接地址 -// */ -// @Getter -// private final String address; -// /** -// * 标签页类型 -// */ -// @Getter -// private final String type; -// private final boolean debug; -// private final String websocketUrl; -// private final AtomicInteger curId; -// /** -// * 会话返回值 -// */ -// private final AtomicReference webSocketMsg = new AtomicReference<>(null); -// private final Thread recvThread; -// private final Thread handleEventThread; -// @Getter -// -// private final AtomicBoolean stopped; -// private final BlockingQueue> eventQueue; -// private final BlockingQueue> immediateEventQueue; -// private final Map eventHandlers; -// private final Map immediateEventHandlers; -// private final Map>> methodResults; -// /** -// * 创建这个驱动的对象 -// */ -// @Getter -// @Setter -// private Occupant occupant; -// private boolean alertFlag; -// /** -// * 会话驱动 -// */ -// private WebSocketClient ws; -// private Thread handleImmediateEventThread; -// -// public Driver(String tabId, String tabType, String address) { -// this(tabId, tabType, address, null); -// } -// -// /** -// * 驱动 -// * -// * @param tabId 标签id -// * @param tabType 标签页类型 -// * @param address 浏览器连接地址 -// */ -// public Driver(String tabId, String tabType, String address, Occupant occupant) { -// this.id = tabId; -// this.address = address; -// this.type = tabType; -// this.occupant = occupant; -// this.debug = true; -// this.alertFlag = false; -// this.websocketUrl = "ws://" + address + "/devtools/" + tabType + "/" + tabId; -// this.curId = new AtomicInteger(0); -// this.ws = null; -// -// -// this.recvThread = new Thread(this::recvLoop); -// this.handleEventThread = new Thread(this::handleEventLoop); -// this.recvThread.setDaemon(true); -// this.handleEventThread.setDaemon(true); -// this.handleImmediateEventThread = null; -// -// this.stopped = new AtomicBoolean(); -// -// this.eventHandlers = new ConcurrentHashMap<>(); -// this.immediateEventHandlers = new ConcurrentHashMap<>(); -// this.methodResults = new ConcurrentHashMap<>(); -// this.eventQueue = new LinkedBlockingQueue<>(); -// this.immediateEventQueue = new LinkedBlockingQueue<>(); -// start(); -// } -// -// /** -// * 发送信息到浏览器,并返回浏览器返回的信息 -// * -// * @param message 发送给浏览器的数据 -// * @param timeout 超时时间,为null表示无时间 -// * @return 浏览器返回的数据 -// */ -// private JSONObject send(Map message, double timeout) { -// message = new HashMap<>(message); -// int wsId = curId.incrementAndGet(); -// message.put("id", wsId); -// String messageJson = JSON.toJSONString(message); -// -// if (this.debug) System.out.println("发->" + messageJson); -// //计算等待时间 -// long endTime = (long) (System.currentTimeMillis() + timeout * 1000L); -// LinkedBlockingQueue> value = new LinkedBlockingQueue<>(); -// methodResults.put(wsId, value); -// try { -// ws.send(messageJson); -// if (timeout == 0) { -// methodResults.remove(wsId); -// return new JSONObject(Map.of("id", wsId, "result", Map.of())); -// } -// } catch (WebsocketNotConnectedException e) { -// e.printStackTrace(); -// methodResults.remove(wsId); -// return new JSONObject(Map.of("error", Map.of("message", "connection disconnected"), "type", "connection_error")); -// } -// int i = 5; -// long endTimes = System.currentTimeMillis() + 1000L; -// while (!stopped.get()) { -// try { -// Map result = methodResults.get(wsId).poll(10_000, TimeUnit.MILLISECONDS); -// if (result == null && System.currentTimeMillis() < endTimes) continue; -// if (result == null && i > 0 && System.currentTimeMillis() > endTimes) { -// i--; -// endTimes = System.currentTimeMillis() + 1000L; -// System.out.println("超时丢包:->" + messageJson); -// ws.send(messageJson); -// continue; -// } -// methodResults.remove(wsId); -// if (result == null) throw new NullPointerException(); -// return new JSONObject(result); -// } catch (InterruptedException | NullPointerException | IllegalArgumentException e) { -//// e.printStackTrace(); -// String string = message.get("method").toString(); -// if (alertFlag && string.startsWith("Input.") || string.startsWith("Runtime.")) { -// return new JSONObject(Map.of("error", Map.of("message", "alert exists."), "type", "alert_exists")); -// } -// if (timeout > 0 && System.currentTimeMillis() > endTime) { -// methodResults.remove(wsId); -// return alertFlag ? new JSONObject(Map.of("error", Map.of("message", "alert exists."), "type", "alert_exists")) : new JSONObject(Map.of("error", Map.of("message", "timeout"), "type", "timeout")); -// } -// } -// } -// -// return new JSONObject(Map.of("error", Map.of("message", "connection disconnected"), "type", "connection_error")); -// } -// -// /** -// * 接收浏览器信息的守护线程方法 -// */ -// private void recvLoop() { -// while (!stopped.get()) { -// JSONObject msg; -// try { -// String andSet = webSocketMsg.getAndSet(null); -// if (andSet != null) { -// msg = JSONObject.parseObject(andSet); -// } else continue; -// } catch (Exception e) { -// if (stop()) return; -// return; -// -// } -// if (this.debug) System.out.println("<-收" + msg); -// -// if (msg.containsKey("method")) { -// if (msg.getString("method").startsWith("Page.javascriptDialog")) { -// alertFlag = msg.getString("method").endsWith("Opening"); -// } -// MyRunnable function = immediateEventHandlers.get(msg.getString("method")); -// if (function != null) { -// this.handleImmediateEvent(function, msg.getOrDefault("params", new HashMap<>())); -// } else { -// try { -// eventQueue.put(msg); -// } catch (InterruptedException e) { -// throw new RuntimeException(e); -// } -// } -// } else { -// int i = 1000; -// Integer integer = msg.getInteger("id"); -// while (i-- > 0 && integer != null && !methodResults.containsKey(integer)) { -// try { -// Thread.sleep(10); -// } catch (InterruptedException e) { -// throw new RuntimeException(e); -// } -// } -// if (methodResults.containsKey(integer)) { -// try { -// methodResults.get(integer).put(msg); -// } catch (InterruptedException e) { -// throw new RuntimeException(e); -// } -// } else if (this.debug) { -// System.out.println("未知错误->" + msg); -// -// } -// } -// -// } -// } -// -// /** -// * 当接收到浏览器信息,执行已绑定的方法 -// */ -// private void handleEventLoop() { -// while (!stopped.get()) { -// Map event; -// try { -// event = eventQueue.poll(1, TimeUnit.SECONDS); -// } catch (InterruptedException e) { -// continue; -// } -// -// if (event != null) { -// MyRunnable function = eventHandlers.get(event.get("method").toString()); -// if (function != null) { -// function.setMessage(event.get("params")); -// function.run(); -// } -// } -// this.eventQueue.poll(); -// -// } -// } -// -// private void handleImmediateEventLoop() { -// while (!stopped.get() && !immediateEventQueue.isEmpty()) { -// Map event; -// try { -// event = immediateEventQueue.poll(1, TimeUnit.SECONDS); -// } catch (InterruptedException e) { -// e.printStackTrace(); -// continue; -// } -// if (event != null) { -// MyRunnable function = immediateEventHandlers.get(event.get("method").toString()); -// if (function != null) { -// function.setMessage(event.get("params")); -// function.run(); -// } -// } -// -// } -// } -// -// /** -// * 处理立即执行的动作 -// * -// * @param function 要运行下方法 -// * @param params 方法参数 -// */ -// private void handleImmediateEvent(MyRunnable function, Object params) { -// Map func = new HashMap<>(); -// func.put("method", function); -// func.put("params", params); -// immediateEventQueue.add(func); -// -// if (handleImmediateEventThread == null || !handleImmediateEventThread.isAlive()) { -// handleImmediateEventThread = new Thread(this::handleImmediateEventLoop); -// handleImmediateEventThread.setDaemon(true); -// handleImmediateEventThread.start(); -// } -// } -// -// /** -// * 执行cdp方法 -// * -// * @param method 方法 -// * @return 执行结果 -// */ -// public Object run(String method) { -// return run(method, new HashMap<>()); -// } -// -// /** -// * 执行cdp方法 -// * -// * @param method 方法 -// * @param params 参数 -// * @return 执行结果 -// */ -// public Object run(String method, Map params) { -// if (stopped.get()) return Map.of("error", "connection disconnected", "type", "connection_error"); -// params = new HashMap<>(params); -// Object timeout1 = params.remove("_timeout"); -// double timeout = timeout1 != null ? Float.parseFloat(timeout1.toString()) : 30.0; -// -// JSONObject result = this.send(Map.of("method", method, "params", params), timeout); -// if (!result.containsKey("result") && result.containsKey("error")) { -// HashMap map = new HashMap<>(); -// map.put("error", result.getJSONObject("error").get("message")); -// map.put("type", result.getOrDefault("type", "call_method_error")); -// map.put("method", method); -// map.put("args", params); -// map.put("timeout", timeout); -// return JSON.toJSONString(map); -// } else { -// return JSON.toJSONString(result.get("result")); -// } -// } -// -// -// /** -// * 启动连接 -// */ -// private void start() { -// this.stopped.set(false); -// try { -// ws = new WebSocketClient(new URI(websocketUrl)) { -// @Override -// public void onOpen(ServerHandshake handshakeData) { -// // 处理 WebSocket 打开事件 -// } -// -// @Override -// public void onMessage(String message) { -// //处理返回数据 -// webSocketMsg.set(message); -// } -// -// -// @Override -// public void onClose(int code, String reason, boolean remote) { -// -// System.out.println("关闭" + reason); -// // 关闭事件处理 -// stop(); -// } -// -// @Override -// public void onError(Exception ex) { -// System.out.println("错误" + ex.getMessage()); -// // 错误事件处理 -//// stop(); -// } -// }; -// ws.setConnectionLostTimeout(60); -// ws.connect(); -// //需要睡0.1秒让其等待 -// while (ws != null && !ws.getReadyState().equals(ReadyState.OPEN)) { -// Thread.sleep(10); -// } -// recvThread.start(); -// handleEventThread.start(); -// } catch (Exception e) { -// e.printStackTrace(); -// stop(); -// } -// } -// -// /** -// * 中断连接 -// */ -// public boolean stop() { -// stop1(); -// while (this.recvThread.isAlive() || this.handleEventThread.isAlive()) { -// try { -// Thread.sleep(100); -// } catch (InterruptedException e) { -// throw new RuntimeException(e); -// } -// } -// return true; -// } -// -// /** -// * 中断连接 -// */ -// private void stop1() { -// if (stopped.get()) { -// return; -// } -// stopped.set(true); -// if (ws != null) { -// ws.close(); -// ws = null; -// } -// -// try { -// while (!eventQueue.isEmpty()) { -// Map
, T extends BaseElement, ?>> extends BaseParser { - private final P owner; - - public BaseElement(P page) { - this.owner = page; - this.setType("BaseElement"); - } - - /** - * @return 返回元素标签名 - */ - public abstract String tag(); - - /** - * 返回上面某一级父元素 用查询语法定位 - * - * @param by 查询选择器 - * @return 上级元素对象 - */ - public T parent(By by) { - return parent(by, 1); - } - - /** - * 返回上面某一级父元素 用查询语法定位 - * - * @param by 查询选择器 - * @param index 选择第几个结果 - * @return 上级元素对象 - */ - public abstract T parent(By by, Integer index); - - /** - * 获取上级父元素 - * - * @return 上级元素对象 - */ - public T parent() { - return parent(1); - } - - /** - * 返回上面某一级父元素,指定层数 - * - * @param level 第几级父元素 - * @return 上级元素对象 - */ - public abstract T parent(Integer level); - - /** - * 获取指定定位的第一个父元素 - * - * @param loc 定位语法 - * @return 上级元素对象 - */ - public T parent(String loc) { - return parent(loc, 1); - } - - /** - * 返回上面某一级父元素 用查询语法定位 - * - * @param loc 定位符 - * @param index 选择第几个结果 - * @return 上级元素对象 - */ - public abstract T parent(String loc, Integer index); - - - /** - * 返回当前元素后面一个符合条件的同级元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param by 用于筛选的查询语法 - * @return 兄弟元素或节点文本 - */ - public T next(By by) { - return next(by, 1); - } - - /** - * 返回当前元素后面一个符合条件的同级元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param by 用于筛选的查询语法 - * @param index 第几个查询结果,0开始 - * @return 兄弟元素或节点文本 - */ - public T next(By by, Integer index) { - return next(by, index, null); - } - - /** - * 返回当前元素后面一个符合条件的同级元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param by 用于筛选的查询语法 - * @param index 第几个查询结果,0开始 - * @param timeout 查找节点的超时时间(秒) - * @return 兄弟元素或节点文本 - */ - public T next(By by, Integer index, Double timeout) { - return next(by, index, timeout, true); - } - - /** - * 返回当前元素后面一个符合条件的同级元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param by 用于筛选的查询语法 - * @param index 第几个查询结果,0开始 - * @param timeout 查找节点的超时时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 兄弟元素或节点文本 - */ - public abstract T next(By by, Integer index, Double timeout, Boolean eleOnly); - - /** - * 返回当前元素后面一个符合条件的同级元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @return 兄弟元素或节点文本 - */ - public T next() { - return next(""); - } - - /** - * 返回当前元素后面一个符合条件的同级元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param loc 用于筛选的查询语法 - * @return 兄弟元素或节点文本 - */ - public T next(String loc) { - return next(loc.isEmpty() ? null : loc, 1); - } - - /** - * 返回当前元素后面一个符合条件的同级元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param loc 用于筛选的查询语法 - * @param index 第几个查询结果,0开始 - * @return 兄弟元素或节点文本 - */ - public T next(String loc, Integer index) { - return next(loc, index, null); - } - - /** - * 返回当前元素后面一个符合条件的同级元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param loc 用于筛选的查询语法 - * @param index 第几个查询结果,0开始 - * @param timeout 查找节点的超时时间(秒) - * @return 兄弟元素或节点文本 - */ - public T next(String loc, Integer index, Double timeout) { - return next(loc, index, timeout, true); - } - - /** - * 返回当前元素后面一个符合条件的同级元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param loc 用于筛选的查询语法 - * @param index 第几个查询结果,0开始 - * @param timeout 查找节点的超时时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 兄弟元素或节点文本 - */ - public abstract T next(String loc, Integer index, Double timeout, Boolean eleOnly); - - /** - * 返回当前元素后面一个符合条件的同级元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param index 第几个查询结果,0开始 - * @return 兄弟元素或节点文本 - */ - - public T next(Integer index) { - return next(index, null); - } - - /** - * 返回当前元素后面一个符合条件的同级元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param index 第几个查询结果,0开始 - * @param timeout 查找节点的超时时间(秒) - * @return 兄弟元素或节点文本 - */ - public T next(Integer index, Double timeout) { - return next(index, timeout, true); - } - - /** - * 返回当前元素后面一个符合条件的同级元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param index 第几个查询结果,0开始 - * @param timeout 查找节点的超时时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 兄弟元素或节点文本 - */ - public abstract T next(Integer index, Double timeout, Boolean eleOnly); - - - @Override - public List _ele(By by, Double timeout, Integer index, Boolean relative, Boolean raiseErr, String method) { - return __ele(this.findElements(by, timeout, index, relative, raiseErr), by.getValue(), index, raiseErr, method); - } - - @Override - public List _ele(String str, Double timeout, Integer index, Boolean relative, Boolean raiseErr, String method) { - return __ele(this.findElements(str, timeout, index, relative, raiseErr), str, index, raiseErr, method); - } - - private List __ele(List elements, String str, Integer index, Boolean raiseErr, String method) { - //如果index为空则说明是查找多元素,如果不为空,则说明是单元素查找,直接判断是否为空,如果不为空则说明单元素找到了 - if (index == null || !elements.isEmpty()) return elements; - //如果是单元素,是否抛出异常 - if (Settings.raiseWhenEleNotFound || raiseErr != null && raiseErr) { - Map locOrStr = new HashMap<>(); - locOrStr.put("loc_or_str", str); - locOrStr.put("index", index); - throw new ElementNotFoundError(null, method, locOrStr); - } - //如果不抛出异常则直接创建个空的 - return new ArrayList<>(); - } - -} diff --git a/java/src/main/java/com/ll/DrissonPage/base/BasePage.java b/java/src/main/java/com/ll/DrissonPage/base/BasePage.java deleted file mode 100644 index f63d502..0000000 --- a/java/src/main/java/com/ll/DrissonPage/base/BasePage.java +++ /dev/null @@ -1,300 +0,0 @@ -package com.ll.DrissonPage.base; - -import com.alibaba.fastjson.JSONObject; -import com.ll.DownloadKit.DownloadKit; -import com.ll.DrissonPage.error.extend.ElementNotFoundError; -import com.ll.DrissonPage.functions.Settings; -import lombok.Getter; -import lombok.Setter; -import okhttp3.Cookie; - -import java.nio.file.InvalidPathException; -import java.nio.file.Path; -import java.nio.file.Paths; -import java.util.ArrayList; -import java.util.HashMap; -import java.util.List; -import java.util.Map; - -/** - * 页面类的基类 - * - * @author 陆 - * @address click - */ -//页面类的基类 -public abstract class BasePage> extends BaseParser { - protected String url = null; - /** - * 查找元素时等待的秒数 - */ - private Double timeout = 10.0; - /** - * 当前访问的url有效性 - */ - @Setter - private Boolean urlAvailable = null; - /** - * 重试次数 - */ - @Setter - @Getter - private Integer retryTimes = 3; - /** - * 重试间隔 - */ - @Setter - @Getter - private Double retryInterval = 2.0; - /** - * 默认下载路径 - */ - @Setter - private String downloadPath = null; - /** - * 下载器对象 - */ - @Getter - private DownloadKit downloadKit = null; - /** - * ele返回值 - */ - @Setter - private Boolean noneEleReturnValue = Boolean.FALSE; - @Setter - private Object noneEleValue = null; - - public BasePage() { - this.setType("BasePage"); - } - - /** - * 返回网页的标题title - */ - public String title() { - T title = this._ele("xpath://title", null, null, null, false, "title").get(0); - return title instanceof DrissionElement ? ((DrissionElement, ?>) title).text() : null; - } - - /** - * 返回查找元素时等待的秒数 - */ - public Double timeout() { - return this.timeout; - } - - /** - * 设置查找元素时等待的秒数 - * - * @param timeout 秒 - */ - public void setTimeout(Double timeout) { - if (timeout != null && timeout >= 0) this.timeout = timeout; - } - - - /** - * 连接前的准备 - * - * @param url 要访问的url - * @param retry 重试次数 - * @param interval 重试间隔 - * @return 重试次数和间隔 - */ - protected BeforeConnect beforeConnect(String url, Integer retry, Double interval) { - boolean isFile = false; - - try { - if (Paths.get(url).toFile().exists() || (!url.contains("://") && !url.contains(":\\\\"))) { - Path p = Paths.get(url); - if (p.toFile().exists()) { - url = p.toAbsolutePath().toString(); - isFile = true; - } - } - } catch (InvalidPathException ignored) { - - } - this.url = url; - retry = retry == null ? this.getRetryTimes() : retry; - interval = interval == null ? this.getRetryInterval() : interval; - return new BeforeConnect(retry, interval, isFile); - } - - /** - * @return 返回当前访问的url有效性 - */ - public Boolean urlAvailable() { - return this.urlAvailable; - } - - /** - * @return 返回默认下载路径 - */ - public String downloadPath() { - return this.downloadPath; - } - - /** - * 返回下载器对象 - */ - public DownloadKit download() { - if (this.downloadKit == null) this.downloadKit = new DownloadKit(this.downloadPath, null, null, this); - return this.downloadKit; - } - - //---------------------------------------------------------------------------------------------------------------------- - - /** - * @return 返回当前访问url - */ - public abstract String url(); - /** - * @return 用于被WebPage覆盖 - */ - protected String browserUrl() { - return this.url(); - } - - public abstract JSONObject json(); - - /** - * @return 返回user agent - */ - public abstract String userAgent(); - - /** - * 返回cookies - */ - public List cookies() { - return cookies(false); - } - - /** - * 返回cookies - * - * @param asMap 为True时返回由{name: value}键值对组成的map,为false时返回list且allInfo无效 - */ - public List cookies(boolean asMap) { - return cookies(asMap, false); - } - - /** - * 返回cookies - * - * @param asMap 为True时返回由{name: value}键值对组成的map,为false时返回list且allInfo无效 - * @param allDomains 是否返回所有域的cookies - */ - public List cookies(boolean asMap, boolean allDomains) { - return cookies(asMap, allDomains, false); - } - - /** - * 返回cookies - * - * @param asMap 为True时返回由{name: value}键值对组成的map,为false时返回list且allInfo无效 - * @param allDomains 是否返回所有域的cookies - * @param allInfo 是否返回所有信息,为False时只返回name、value、domain - */ - public abstract List cookies(boolean asMap, boolean allDomains, boolean allInfo); - - public Boolean get(String url) { - return get(url, false); - } - - - public Boolean get(String url, boolean showErrMsg) { - return get(url, showErrMsg, retryTimes, retryInterval, timeout); - } - - public Boolean get(String url, boolean showErrMsg, Integer retry, Double interval, Double timeout) { - return get(url, showErrMsg, retry, interval, timeout, null); - } - - /** - * 用get请求跳转到url,可输入文件路径 - * - * @param url 目标url - * @param showErrMsg 是否显示和抛出异常 - * @param retry 重试次数,为null时使用页面对象retryTimes属性值 - * @param interval 重试间隔(秒),为null时使用页面对象retryInterval属性值 - * @param timeout 连接超时时间(秒),为null时使用页面对象timeouts.pageLoad属性值 - * @param params 连接参数 s模式专用 - * @return url是否可用 - */ - public abstract Boolean get(String url, boolean showErrMsg, Integer retry, Double interval, Double timeout, Map params); - - /** - * @param by 查询元素 - * @param timeout 查找超时时间(秒) - * @param index 获取第几个,从0开始,可传入负数获取倒数第几个 如果是null则是返回全部 - * @param relative WebPage用的表示是否相对定位的参数 - * @param raiseErr 找不到元素是是否抛出异常,为null时根据全局设置 - * @param method 方法名称 - * @return 元素对象组成的列表 - */ - @Override - public List _ele(By by, Double timeout, Integer index, Boolean relative, Boolean raiseErr, String method) { - Map map = new HashMap<>(); - map.put("str", ""); - map.put("index", index); - if (by == null) throw new ElementNotFoundError(null, method, map); - List elements = this.findElements(by, timeout, index, relative, raiseErr); - //如果index为空则说明是查找多元素,如果不为空,则说明是单元素查找,直接判断是否为空,如果不为空则说明单元素找到了 - if (index == null || !elements.isEmpty()) return elements; - //如果是单元素,是否抛出异常 - if (Settings.raiseWhenEleNotFound || raiseErr != null && raiseErr) - throw new ElementNotFoundError(null, method, map); - //如果不抛出异常则直接创建个空的 - return new ArrayList<>(); - } - - /** - * @param loc 定位符 - * @param timeout 查找超时时间(秒) - * @param index 获取第几个,从0开始,可传入负数获取倒数第几个 如果是null则是返回全部 - * @param relative WebPage用的表示是否相对定位的参数 - * @param raiseErr 找不到元素是是否抛出异常,为null时根据全局设置 - * @param method 方法名称 - * @return 元素对象组成的列表 - */ - @Override - public List _ele(String loc, Double timeout, Integer index, Boolean relative, Boolean raiseErr, String method) { - Map map = new HashMap<>(); - map.put("str", ""); - map.put("index", index); - if (loc == null) throw new ElementNotFoundError(null, method, map); - List elements = this.findElements(loc, timeout, index, relative, raiseErr); - //如果index为空则说明是查找多元素,如果不为空,则说明是单元素查找,直接判断是否为空,如果不为空则说明单元素找到了 - if (index == null || !elements.isEmpty()) return elements; - //如果是单元素,是否抛出异常 - if (Settings.raiseWhenEleNotFound || raiseErr != null && raiseErr) - throw new ElementNotFoundError(null, method, map); - //如果不抛出异常则直接创建个空的 - return new ArrayList<>(); - } - /** - * 执行元素查找 - * - * @param by 查询元素 - * @param timeout 查找超时时间(秒) - * @param index 获取第几个,从1开始,可传入负数获取倒数第几个 如果是null则是返回全部 - * @param relative WebPage用的表示是否相对定位的参数 - * @param raiseErr 找不到元素是是否抛出异常,为null时根据全局设置 - * @return 元素对象组成的列表 - */ - protected abstract List findElements(By by, Double timeout, Integer index, Boolean relative, Boolean raiseErr); - - - /** - * 执行元素查找 - * - * @param loc 定位符 - * @param timeout 查找超时时间(秒) - * @param index 获取第几个,从1开始,可传入负数获取倒数第几个 如果是null则是返回全部 - * @param relative WebPage用的表示是否相对定位的参数 - * @param raiseErr 找不到元素是是否抛出异常,为null时根据全局设置 - * @return 元素对象组成的列表 - */ - protected abstract List findElements(String loc, Double timeout, Integer index, Boolean relative, Boolean raiseErr); -} diff --git a/java/src/main/java/com/ll/DrissonPage/base/BaseParser.java b/java/src/main/java/com/ll/DrissonPage/base/BaseParser.java deleted file mode 100644 index 6521aa7..0000000 --- a/java/src/main/java/com/ll/DrissonPage/base/BaseParser.java +++ /dev/null @@ -1,235 +0,0 @@ -package com.ll.DrissonPage.base; - -import com.ll.DrissonPage.element.SessionElement; -import com.ll.DrissonPage.error.extend.ElementNotFoundError; -import lombok.Getter; -import lombok.Setter; - -import java.util.List; -import java.util.Map; - -/** - * 所有页面、元素类的基类 - * - * @author 陆 - * @address click - */ - -@Getter -public abstract class BaseParser> { - @Setter - private String type; - - - /** - * 返回当前元素下级符合条件的一个元素、属性或节点文本 - * - * @param by 查询元素 - */ - public T ele(By by) { - return ele(by, 1); - } - - /** - * 返回当前元素下级符合条件的一个元素、属性或节点文本 - * - * @param by 查询元素 - * @param index 获取第几个元素,下标从1开始可传入负数获取倒数第几个 - */ - public T ele(By by, int index) { - return ele(by, index, null); - } - - /** - * 返回当前元素下级符合条件的一个元素、属性或节点文本 - * - * @param by 查询元素 - * @param timeout 查找元素超时时间(秒),默认与元素所在页面等待时间一致 - */ - public T ele(By by, Double timeout) { - return ele(by, 1, timeout); - } - - /** - * 返回当前元素下级符合条件的一个元素、属性或节点文本 - * - * @param by 查询元素 - * @param index 获取第几个元素,下标从1开始可传入负数获取倒数第几个 - * @param timeout 查找元素超时时间(秒),默认与元素所在页面等待时间一致 - */ - public T ele(By by, int index, Double timeout) { - try { - return this._ele(by, timeout, index, null, null, "ele()").get(0); - } catch (IndexOutOfBoundsException e) { - Map map = new java.util.HashMap<>(); - map.put("by", by); - map.put("index", index); - new ElementNotFoundError("ele()", map).printStackTrace(); - return null; - } - } - - /** - * 获取单个元素 - * - * @param loc 参数,字符串 - */ - public T ele(String loc) { - return ele(loc, 1); - } - - /** - * 获取单个元素 - * - * @param loc 参数,字符串 - * @param index 获取第几个元素,下标从0开始可传入负数获取倒数第几个 - */ - public T ele(String loc, int index) { - return ele(loc, index, null); - } - - - /** - * 返回当前元素下级符合条件的一个元素、属性或节点文本 - * - * @param loc 参数,字符串 - * @param timeout 查找元素超时时间(秒),默认与元素所在页面等待时间一致 - */ - public T ele(String loc, Double timeout) { - return ele(loc, 1, timeout); - } - - /** - * 获取单个元素 - * - * @param loc 参数,字符串 - * @param index 获取第几个元素,下标从0开始可传入负数获取倒数第几个 - * @param timeout 查找元素超时时间(秒),默认与元素所在页面等待时间一致 - */ - public T ele(String loc, int index, Double timeout) { - try { - List ts = this._ele(loc, timeout, index, null, null, "ele()"); - if (ts == null) return null; - return ts.get(0); - } catch (IndexOutOfBoundsException e) { - Map map = new java.util.HashMap<>(); - map.put("loc", loc); - map.put("index", index); - new ElementNotFoundError("ele()", map).printStackTrace(); - return null; - } - } - - - public List eles(By by) { - return eles(by, null); - } - - public List eles(By by, Double timeout) { - return this._ele(by, timeout, null, null, null, null); - } - - public List eles(String loc) { - return eles(loc, null); - } - - public List eles(String loc, Double timeout) { - return this._ele(loc, timeout, null, null, null, null); - } - - /** - * 获取当前页面数据 - */ - public abstract String html(); - - /** - * @param by 查询元素 - * @return 元素对象 - */ - public SessionElement sEle(By by) { - return sEle(by, 1); - } - - /** - * @param by 查询元素 - * @param index 获取第几个,从1开始,可传入负数获取倒数第几个 - * @return 元素对象 - */ - public abstract SessionElement sEle(By by, Integer index); - - /** - * @param loc 定位符 - * @return 元素对象 - */ - public SessionElement sEle(String loc) { - return sEle(loc, 1); - } - - /** - * @param loc 定位符 - * @param index 获取第几个,从1开始,可传入负数获取倒数第几个 - * @return 元素对象 - */ - public abstract SessionElement sEle(String loc, Integer index); - - /** - * @param by 查询元素 - * @return 元素对象组成的列表 - */ - public abstract List sEles(By by); - - /** - * @param loc 定位符 - * @return 元素对象组成的列表 - */ - public abstract List sEles(String loc); - - - /** - * @param by 查询元素 - * @param timeout 查找超时时间(秒) - * @param index 获取第几个,从0开始,可传入负数获取倒数第几个 如果是null则是返回全部 - * @param relative WebPage用的表示是否相对定位的参数 - * @param raiseErr 找不到元素是是否抛出异常,为null时根据全局设置 - * @param method 方法名称 - * @return 元素对象组成的列表 - */ - protected abstract List _ele(By by, Double timeout, Integer index, Boolean relative, Boolean raiseErr, String method); - - /** - * @param loc 定位符 - * @param timeout 查找超时时间(秒) - * @param index 获取第几个,从0开始,可传入负数获取倒数第几个 如果是null则是返回全部 - * @param relative WebPage用的表示是否相对定位的参数 - * @param raiseErr 找不到元素是是否抛出异常,为null时根据全局设置 - * @param method 方法名称 - * @return 元素对象组成的列表 - */ - protected abstract List _ele(String loc, Double timeout, Integer index, Boolean relative, Boolean raiseErr, String method); - - /** - * 执行元素查找 - * - * @param by 查询元素 - * @param timeout 查找超时时间(秒) - * @param index 获取第几个,从0开始,可传入负数获取倒数第几个 如果是null则是返回全部 - * @param relative WebPage用的表示是否相对定位的参数 - * @param raiseErr 找不到元素是是否抛出异常,为null时根据全局设置 - * @return 元素对象组成的列表 - */ - protected abstract List findElements(By by, Double timeout, Integer index, Boolean relative, Boolean raiseErr); - - - /** - * 执行元素查找 - * - * @param loc 定位符 - * @param timeout 查找超时时间(秒) - * @param index 获取第几个,从0开始,可传入负数获取倒数第几个 如果是null则是返回全部 - * @param relative WebPage用的表示是否相对定位的参数 - * @param raiseErr 找不到元素是是否抛出异常,为null时根据全局设置 - * @return 元素对象组成的列表 - */ - protected abstract List findElements(String loc, Double timeout, Integer index, Boolean relative, Boolean raiseErr); - -} \ No newline at end of file diff --git a/java/src/main/java/com/ll/DrissonPage/base/BeforeConnect.java b/java/src/main/java/com/ll/DrissonPage/base/BeforeConnect.java deleted file mode 100644 index 74a59d6..0000000 --- a/java/src/main/java/com/ll/DrissonPage/base/BeforeConnect.java +++ /dev/null @@ -1,16 +0,0 @@ -package com.ll.DrissonPage.base; - -import lombok.AllArgsConstructor; -import lombok.Getter; - -/** - * @author 陆 - * @address click - */ -@AllArgsConstructor -@Getter -public class BeforeConnect { - private Integer retry; - private Double interval; - private boolean isFile; -} diff --git a/java/src/main/java/com/ll/DrissonPage/base/Browser.java b/java/src/main/java/com/ll/DrissonPage/base/Browser.java deleted file mode 100644 index 8a6d5fa..0000000 --- a/java/src/main/java/com/ll/DrissonPage/base/Browser.java +++ /dev/null @@ -1,470 +0,0 @@ -package com.ll.DrissonPage.base; - -import com.alibaba.fastjson.JSON; -import com.alibaba.fastjson.JSONArray; -import com.alibaba.fastjson.JSONObject; -import com.ll.DrissonPage.config.ChromiumOptions; -import com.ll.DrissonPage.error.extend.PageDisconnectedError; -import com.ll.DrissonPage.functions.Tools; -import com.ll.DrissonPage.page.ChromiumPage; -import com.ll.DrissonPage.page.ChromiumBase; -import com.ll.DrissonPage.units.downloader.DownloadManager; -import lombok.Getter; - -import java.io.BufferedReader; -import java.io.File; -import java.io.IOException; -import java.io.InputStreamReader; -import java.nio.file.FileVisitOption; -import java.nio.file.Files; -import java.nio.file.Path; -import java.nio.file.Paths; -import java.util.*; -import java.util.concurrent.ConcurrentHashMap; -import java.util.stream.Collectors; -import java.util.stream.Stream; - -/** - * 浏览器 - * - * @author 陆 - * @address click - */ - -public class Browser implements Occupant { - public static final String __ERROR__ = "error"; - private static final Map BROWSERS = new ConcurrentHashMap<>(); - @Getter - private final ChromiumBase page; - private final Map drivers; - private final Map> allDrivers; - @Getter - private BrowserDriver driver; - @Getter - private String id; - @Getter - private String address; - @Getter - private Map frames; - @Getter - private DownloadManager dlMgr; - /** - * 浏览器进程id - */ - @Getter - private Integer processId; - private boolean connected; - - /** - * 浏览器 - * - * @param address 浏览器地址 - * @param browserId 浏览器id - * @param page ChromiumPage对象 - */ - private Browser(String address, String browserId, ChromiumBase page) { - this.page = page; - this.address = address; - this.driver = BrowserDriver.getInstance(browserId, "browser", address, this); - this.id = browserId; - this.frames = new HashMap<>(); - this.drivers = new HashMap<>(); - this.allDrivers = new HashMap<>(); - this.connected = false; - this.processId = null; - JSONArray processInfoArray = JSON.parseObject(runCdp("SystemInfo.getProcessInfo")).getJSONArray("processInfo"); - if (processInfoArray == null) processInfoArray = new JSONArray(); - for (Object processInfoObject : processInfoArray) { - JSONObject processInfo = JSON.parseObject(processInfoObject.toString()); - if ("browser".equals(processInfo.getString("type"))) { - this.processId = processInfo.getInteger("id"); - break; - } - } - this.runCdp("Target.setDiscoverTargets", Map.of("discover", true)); - driver.setCallback("Target.targetDestroyed", new MyRunnable() { - @Override - public void run() { - onTargetDestroyed(getMessage()); - } - }); - driver.setCallback("Target.targetCreated", new MyRunnable() { - @Override - public void run() { - onTargetCreated(getMessage()); - } - }); - } - - /** - * 单例模式 - * - * @param address 浏览器地址 - * @param browserId 浏览器id - * @param page ChromiumPage对象 - */ - public static Browser getInstance(String address, String browserId, ChromiumBase page) { - return BROWSERS.computeIfAbsent(browserId, key -> new Browser(address, browserId, page)); - } - - /** - * 获取对应tab id的Driver - * - * @param tabId 标签页id - * @return Driver对象 - */ - public Driver getDriver(String tabId) { - return getDriver(tabId, null); - } - - /** - * 获取对应tab id的Driver - * - * @param tabId 标签页id - * @param occupant 使用该驱动的对象 - * @return Driver对象 - */ - public Driver getDriver(String tabId, Occupant occupant) { - Driver driver = Objects.requireNonNullElseGet(drivers.remove(tabId), () -> new Driver(tabId, "page", this.getAddress(), occupant)); - HashSet value = new HashSet<>(); - value.add(driver); - this.allDrivers.put(tabId, value); - return driver; - } - - /** - * 标签页创建时执行 - * - * @param message 回调参数 - */ - private void onTargetCreated(Object message) { - try { - JSONObject jsonObject = JSON.parseObject(message.toString()); - String type = jsonObject.getJSONObject("targetInfo").getString("type"); - if ("page".equals(type) || "webview".equals(type) && !jsonObject.getJSONObject("targetInfo").getString("url").startsWith("devtools://")) { - String targetId = jsonObject.getJSONObject("targetInfo").getString("targetId"); - Driver driver = new Driver(targetId, "page", address); - drivers.put(targetId, driver); - this.allDrivers.computeIfAbsent(targetId, k -> new HashSet<>()).add(driver); - } - } catch (Exception e) { - e.printStackTrace(); - } - } - - /** - * 标签页关闭时执行 - * - * @param message 回调参数 - */ - private void onTargetDestroyed(Object message) { - JSONObject jsonObject = JSON.parseObject(message.toString()); - String tabId = jsonObject.getString("targetId"); - if (this.dlMgr != null) this.dlMgr.clearTabInfo(tabId); - if (frames != null) frames.values().removeIf(value -> value.equals(tabId)); - allDrivers.forEach((a, b) -> { - if (a.equals(tabId)) b.forEach(Driver::stop); - }); - allDrivers.remove(tabId); - drivers.forEach((a, b) -> { - if (a.equals(tabId)) b.stop(); - }); - drivers.remove(tabId); - - } - - /** - * 执行page相关的逻辑 - */ - public void connectToPage() { - if (!connected) { - this.dlMgr = new DownloadManager(this); - connected = true; - } - } - - public String runCdp(String cmd) { - return runCdp(cmd, new HashMap<>()); - } - - /** - * 执行Chrome DevTools Protocol语句 - * - * @param cmd 协议项目 - * @param cmdArgs 参数 - * @return 执行的结果 - */ - public String runCdp(String cmd, Map cmdArgs) { - cmdArgs = new HashMap<>(cmdArgs == null ? new HashMap<>() : cmdArgs); - Object ignore = cmdArgs.remove("_ignore"); - String result = driver.run(cmd, cmdArgs).toString(); - JSONObject result1 = JSONObject.parseObject(result); - if (result1.containsKey(__ERROR__)) Tools.raiseError(result1, ignore); - return result; - } - - /** - * 返回标签页数量 - */ - public int tabsCount() { - JSONArray targetInfos = JSON.parseObject(this.runCdp("Target.getTargets")).getJSONArray("targetInfos"); - return (int) targetInfos.stream().filter(targetInfo -> { - JSONObject jsonObject = JSON.parseObject(targetInfo.toString()); - String type = jsonObject.getString("type"); - String url = jsonObject.getString("url"); - return ("page".equals(type) || "webview".equals(type)) && !url.startsWith("devtools://"); - }).count(); - } - - /** - * 返回所有标签页id组成的列表 - */ - public List tabs() { - JSONArray jsonArray = JSON.parseArray(JSONObject.toJSONString(driver.get("http://" + address + "/json"))); - return jsonArray.stream().filter(targetInfo -> { - JSONObject jsonObject = JSON.parseObject(targetInfo.toString()); - String type = jsonObject.getString("type"); - String url = jsonObject.getString("url"); - return ("page".equals(type) || "webview".equals(type)) && !url.startsWith("devtools://"); - }).map(obj -> ((JSONObject) obj).getString("id")).collect(Collectors.toList()); - } - - /** - * 查找符合条件的tab,返回它们的id组成的列表 - * - * @return tab id或tab列表 - */ - public List findTabs() { - return findTabs(null); - } - - /** - * 查找符合条件的tab,返回它们的id组成的列表 - * - * @param title 要匹配title的文本 - * @return tab id或tab列表 - */ - public List findTabs(String title) { - return findTabs(title, null); - } - - /** - * 查找符合条件的tab,返回它们的id组成的列表 - * - * @param title 要匹配title的文本 - * @param url 要匹配url的文本 - * @return tab id或tab列表 - */ - public List findTabs(String title, String url) { - return findTabs(title, url, null); - } - - /** - * 查找符合条件的tab,返回它们的id组成的列表 - * - * @param title 要匹配title的文本 - * @param url 要匹配url的文本 - * @param tabType tab类型,可用列表输入多个 - * @return tab id或tab列表 - */ - public List findTabs(String title, String url, List tabType) { - return findTabs(title, url, tabType, true); - } - - /** - * 查找符合条件的tab,返回它们的id组成的列表 - * - * @param title 要匹配title的文本 - * @param url 要匹配url的文本 - * @param tabType tab类型,可用列表输入多个 - * @param single 是否返回首个结果的id,为False返回所有信息 - * @return tab id或tab列表 - */ - public List findTabs(String title, String url, List tabType, boolean single) { - Object parse = JSON.parse(JSONObject.toJSONString(this.driver.get("http://" + this.address + "/json"))); - JSONArray tabs; - if (parse instanceof String) { - tabs = new JSONArray(List.of(parse)); - } else if (parse instanceof List || parse instanceof Set || parse instanceof String[]) { - tabs = JSON.parseArray(parse.toString()); - } else { - throw new IllegalArgumentException("tab_type类型不对" + parse.toString()); - } - - List result = tabs.stream().filter(targetInfo -> { - JSONObject jsonObject = JSON.parseObject(targetInfo.toString()); - return (title == null || jsonObject.getString("title").contains(title)) && (url == null || jsonObject.getString("url").contains(url)) && (tabType == null || tabType.contains(jsonObject.getString("type"))); - }).map(tab -> ((JSONObject) tab).getString("id")).collect(Collectors.toList()); - return single ? (result.isEmpty() ? null : List.of(result.get(0))) : result; - } - - /** - * 关闭标签页 - * - * @param tabId 标签页id - */ - public void closeTab(String tabId) { - this.onTargetDestroyed(JSON.toJSONString(Map.of("targetId", tabId))); - this.driver.run("Target.closeTarget", Map.of("targetId", tabId)); - } - - /** - * 停止一个Driver - * - * @param driver Driver对象 - */ - public void stopDiver(Driver driver) { - driver.stop(); - Set set = this.allDrivers.get(driver.getId()); - if (set != null) set.remove(driver); - } - - /** - * 使标签页变为活动状态 - * - * @param tabId 标签页id - */ - public void activateTab(String tabId) { - this.runCdp("Target.activateTarget", Map.of("targetId", tabId)); - } - - /** - * 返回浏览器窗口位置和大小信息 - * - * @return 窗口大小字典 - */ - public JSONObject getWindowBounds() { - return getWindowBounds(null); - } - - /** - * 返回浏览器窗口位置和大小信息 - * - * @param tabId 标签页id 不传入默认使用本身的id - */ - public JSONObject getWindowBounds(String tabId) { - return JSON.parseObject(runCdp("Browser.getWindowForTarget", Map.of("targetId", tabId == null || tabId.isEmpty() ? this.id : tabId))).getJSONObject("bounds"); - } - - /** - * 断开重连 - */ - public void reconnect() { - this.driver.stop(); - BrowserDriver.BROWSERS.remove(this.id); - this.driver = BrowserDriver.getInstance(this.id, "browser", this.address, this); - this.runCdp("Target.setDiscoverTargets", Map.of("discover", true)); - this.driver.setCallback("Target.targetDestroyed", new MyRunnable() { - @Override - public void run() { - onTargetDestroyed(getMessage()); - } - }); - this.driver.setCallback("Target.targetCreated", new MyRunnable() { - @Override - public void run() { - onTargetCreated(getMessage()); - } - }); - } - - /** - * 关闭浏览器 - */ - public void quit() { - quit(5.0); - } - - /** - * 关闭浏览器 - * - * @param timeout 等待浏览器关闭超时时间 - */ - public void quit(double timeout) { - quit(timeout, false); - } - - /** - * 关闭浏览器 - * - * @param timeout 等待浏览器关闭超时时间 - * @param force 是否立刻强制终止进程 - */ - public void quit(double timeout, boolean force) { - List pids = JSON.parseArray(this.runCdp("SystemInfo.getProcessInfo")).stream().map(o -> JSON.parseObject(o.toString()).getInteger("id")).collect(Collectors.toList()); - - - for (Set value : this.allDrivers.values()) for (Driver driver1 : value) driver1.stop(); - - if (force) { - for (Integer pid : pids) - try { - ProcessHandle.allProcesses().filter(process -> process.info().command().isPresent()).filter(process -> process.info().command().map(s -> s.contains(Integer.toString(pid))).orElse(false)).forEach(process -> { - if (process.isAlive()) process.destroy(); - }); - } catch (SecurityException ignored) { - } - - } else { - try { - this.runCdp("Browser.close"); - this.driver.stop(); - } catch (PageDisconnectedError e) { - this.driver.stop(); - return; - } - } - - - if (force) { - String[] ipPort = address.split(":"); - if (!("127.0.0.1".equals(ipPort[0]) || "localhost".equals(ipPort[0]))) { - return; - } - Tools.stopProcessOnPort(Integer.parseInt(ipPort[1])); - return; - } - - if (processId != null) { - String txt = System.getProperty("os.name").toLowerCase().contains("win") ? "tasklist | findstr " + processId : "ps -ef | grep " + processId; - long endTime = (long) (System.currentTimeMillis() + timeout * 1000); - while (System.currentTimeMillis() < endTime) { - try (BufferedReader reader = new BufferedReader(new InputStreamReader(new ProcessBuilder(txt.split("\\s+")).start().getInputStream()))) { - try { - if (!reader.readLine().contains(processId.toString())) { - return; - } - } catch (NullPointerException e) { - // Handle null pointer exception, if needed - } - Thread.sleep(100); - } catch (IOException | InterruptedException e) { - e.printStackTrace(); - } - } - } - } - - @Override - public void onDisconnect() { - this.page.onDisconnect(); - BROWSERS.remove(id); - if (page instanceof ChromiumPage) { - ChromiumOptions chromiumOptions = ((ChromiumPage) page).getChromiumOptions(); - if (chromiumOptions.isAutoPort() && chromiumOptions.getUserDataPath() != null) { - Path path = Paths.get(chromiumOptions.getUserDataPath()); - long endTime = System.currentTimeMillis() + 7000; - while (System.currentTimeMillis() < endTime) { - if (!Files.exists(path)) break; - try (Stream walk = Files.walk(path, FileVisitOption.FOLLOW_LINKS)) { - walk.sorted(Comparator.reverseOrder()).map(Path::toFile).forEach(File::delete); - break; - } catch (IOException e) { - e.printStackTrace(); - } - } - } - } - - } -} \ No newline at end of file diff --git a/java/src/main/java/com/ll/DrissonPage/base/BrowserDriver.java b/java/src/main/java/com/ll/DrissonPage/base/BrowserDriver.java deleted file mode 100644 index e983b7e..0000000 --- a/java/src/main/java/com/ll/DrissonPage/base/BrowserDriver.java +++ /dev/null @@ -1,49 +0,0 @@ -package com.ll.DrissonPage.base; - -import com.alibaba.fastjson.JSON; -import com.ll.DrissonPage.utils.CloseableHttpClientUtils; -import org.apache.http.client.methods.HttpGet; - -import java.util.Map; -import java.util.concurrent.ConcurrentHashMap; - -/** - * 浏览器驱动 - * - * @author 陆 - * @address click - */ - -public class BrowserDriver extends Driver { - protected static final Map BROWSERS = new ConcurrentHashMap<>(); - - - private BrowserDriver(String tabId, String tabType, String address, Occupant occupant) { - super(tabId, tabType, address, occupant); - } - - public static BrowserDriver getInstance(String tabId, String tabType, String address, Occupant occupant) { - - return BROWSERS.computeIfAbsent(tabId, key -> new BrowserDriver(tabId, tabType, address, occupant)); - } - - @Override - public String toString() { - return ""; - } - - /** - * 发送请求 - * - * @param url 请求地址 - * @return 返回值可能是List 或者Map - */ - public Object get(String url) { - HttpGet request = new HttpGet(url); - request.setHeader("Connection", "close"); - String text = CloseableHttpClientUtils.sendRequestJson(request); - return JSON.parse(text); - } - - -} \ No newline at end of file diff --git a/java/src/main/java/com/ll/DrissonPage/base/By.java b/java/src/main/java/com/ll/DrissonPage/base/By.java deleted file mode 100644 index 9b9c736..0000000 --- a/java/src/main/java/com/ll/DrissonPage/base/By.java +++ /dev/null @@ -1,91 +0,0 @@ -package com.ll.DrissonPage.base; - -import lombok.Getter; -import lombok.Setter; - -import java.util.Objects; - -/** - * @author 陆 - * @address click - */ - -@Getter -@Setter -public class By { - private BySelect name; - private String value; - - private By(BySelect name, String value) { - - this.name = name; - this.value = value; - } - - public static By NULL() { - return null; - } - - public static By id(String value) { - return new By(BySelect.ID, Objects.requireNonNullElse(value, "Cannot find elements when id is null.")); - } - - public static By className(String value) { - return new By(BySelect.CLASS_NAME, Objects.requireNonNullElse(value, "Cannot find elements when the class name expression is null.")); - } - - public static By tag(String value) { - return new By(BySelect.TAG_NAME, Objects.requireNonNullElse(value, "Cannot find elements when the tag name is null.")); - } - - public static By name(String value) { - return new By(BySelect.NAME, Objects.requireNonNullElse(value, "Cannot find elements when name text is null.")); - } - - public static By css(String value) { - return new By(BySelect.CSS_SELECTOR, Objects.requireNonNullElse(value, "Cannot find elements when the css selector is null.")); - } - - public static By xpath(String value) { - return new By(BySelect.XPATH, Objects.requireNonNullElse(value, "Cannot find elements when the XPath is null.")); - } - - public static By linkText(String value) { - return new By(BySelect.LINK_TEXT, Objects.requireNonNullElse(value, "Cannot find elements when the link text is null.")); - } - - public static By partialLinkText(String value) { - return new By(BySelect.PARTIAL_LINK_TEXT, Objects.requireNonNullElse(value, "Cannot find elements when the partial link text is null.")); - } - - public static By text(String value) { - return new By(BySelect.TEXT, Objects.requireNonNullElse(value, "Cannot find elements when the text is null.")); - } - - public static By partialText(String value) { - return new By(BySelect.PARTIAL_TEXT, Objects.requireNonNullElse(value, "Cannot find elements when the partial text is null.")); - } - - @Override - public boolean equals(Object o) { - if (this == o) return true; - if (!(o instanceof By)) return false; - By by = (By) o; - return Objects.equals(getName().getName(), by.getName().getName()) && Objects.equals(getValue(), by.getValue()); - } - - @Override - public int hashCode() { - return Objects.hash(getName(), getValue()); - } - - @Override - public String toString() { - return "By{" + - "name=" + name.getName() + - ", value='" + value + '\'' + - '}'; - } - - -} diff --git a/java/src/main/java/com/ll/DrissonPage/base/BySelect.java b/java/src/main/java/com/ll/DrissonPage/base/BySelect.java deleted file mode 100644 index 5531afb..0000000 --- a/java/src/main/java/com/ll/DrissonPage/base/BySelect.java +++ /dev/null @@ -1,18 +0,0 @@ -package com.ll.DrissonPage.base; - -import lombok.Getter; - -/** - * @author 陆 - * @address click - */ -@Getter -public enum BySelect { - NAME("name"), ID("id"), CLASS_NAME("class name"), TAG_NAME("tag name"), CSS_SELECTOR("css selector"), - XPATH("xpath"), LINK_TEXT("link text"), PARTIAL_LINK_TEXT("partial link text"), TEXT("text"), PARTIAL_TEXT("partial text"); - private final String name; - - BySelect(String name) { - this.name = name; - } -} diff --git a/java/src/main/java/com/ll/DrissonPage/base/DrissionElement.java b/java/src/main/java/com/ll/DrissonPage/base/DrissionElement.java deleted file mode 100644 index ca607d4..0000000 --- a/java/src/main/java/com/ll/DrissonPage/base/DrissionElement.java +++ /dev/null @@ -1,1270 +0,0 @@ -package com.ll.DrissonPage.base; - -import com.ll.DrissonPage.error.extend.ElementNotFoundError; -import com.ll.DrissonPage.functions.Locator; -import com.ll.DrissonPage.functions.Settings; -import com.ll.DrissonPage.functions.Web; - -import java.util.ArrayList; -import java.util.List; -import java.util.Map; -import java.util.regex.Pattern; -import java.util.stream.Collectors; - -/** - * ChromiumElement 和 SessionElement的基类 但不是ShadowRoot的基类 - * - * @author 陆 - * @address click - */ -public abstract class DrissionElement, T extends DrissionElement, ?>> extends BaseElement { - - public DrissionElement(P page) { - super(page); - this.setType("DrissionElement"); - } - - /** - * 返回href或者src绝对的url - */ - public String link() { - String href = this.attr("href"); - return href == null ? this.attr("src") : href; - } - - /** - * 返回css path路径 - */ - public String cssPath() { - return this.getElePath(ElePathMode.CSS); - } - - /** - * 返回xpath路径 - */ - public String xpath() { - return this.getElePath(ElePathMode.XPATH); - } - - /** - * 返回元素注释文本组成的列表 - */ - public List comments() { - return this.eles("xpath:.//comment()"); - } - - /** - * 返回元素内所有直接子节点的文本,包括元素和文本节点 - * - * @return 文本列表 - */ - public List texts() { - return this.texts(false); - } - - /** - * 返回元素内所有直接子节点的文本,包括元素和文本节点 - * - * @param textNodeOnly 是否只返回文本节点 - * @return 文本列表 - */ - public List texts(boolean textNodeOnly) { - List texts = new ArrayList<>(); - if (textNodeOnly) { - this.eles("xpath:/text()").forEach(a -> texts.add(a.text())); - } else { - this.eles("xpath:/text() | *").forEach(a -> texts.add(a.text())); - } - return texts.stream().filter(text -> text != null && !Pattern.compile("[\r\n\t ]").matcher(text).replaceAll("").isEmpty()).map(text -> Web.formatHtml(text.trim().replaceAll("[\r\n]", ""))).collect(Collectors.toList()); - } - - /** - * 返回上面某一级父元素 用查询语法定位 - * - * @param by 查询选择器 - * @param index 选择第几个结果 - * @return 上级元素对象 - */ - @Override - public T parent(By by, Integer index) { - by = Locator.getLoc(by, true, false); - if (!by.getName().equals(BySelect.CSS_SELECTOR)) - throw new IllegalArgumentException("此css selector语法不受支持,请换成xpath"); - String loc = "xpath:./ancestor::" + by.getValue().replaceAll("^[.\\s/]+", "") + "[" + index + "]"; - return _ele(loc, null, 1, true, false, "parent()").get(0); - } - - /** - * 返回上面某一级父元素,指定层数 - * - * @param level 第几级父元素 - * @return 上级元素对象 - */ - @Override - public T parent(Integer level) { - String loc = "xpath:./ancestor::*[" + level + "]"; - return _ele(loc, null, 1, true, false, "parent()").get(0); - } - - /** - * 返回上面某一级父元素 用查询语法定位 - * - * @param loc 定位符 - * @param index 选择第几个结果 - * @return 上级元素对象 - */ - @Override - public T parent(String loc, Integer index) { - By loc1 = Locator.getLoc(loc, true, false); - if (!loc1.getName().equals(BySelect.CSS_SELECTOR)) - throw new IllegalArgumentException("此css selector语法不受支持,请换成xpath"); - loc = "xpath:./ancestor::" + loc1.getValue().replaceAll("^[.\\s/]+", "") + "[" + index + "]"; - return _ele(loc, null, 1, true, false, "parent()").get(0); - } - - /** - * 返回直接子元素元素或节点组成的列表,可用查询语法筛选 - * - * @param index 第几个查询结果,1开始 - * @return 直接子元素或节点文本组成的列表 - */ - public T child(Integer index) { - return child(index, null); - } - - /** - * 返回直接子元素元素或节点组成的列表,可用查询语法筛选 - * - * @param index 第几个查询结果,1开始 - * @param timeout 查找节点的超时时间 - * @return 直接子元素或节点文本组成的列表 - */ - public T child(Integer index, Double timeout) { - return child(index, timeout, true); - } - - /** - * 返回直接子元素元素或节点组成的列表,可用查询语法筛选 - * - * @param index 第几个查询结果,1开始 - * @param timeout 查找节点的超时时间 - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 直接子元素或节点文本组成的列表 - */ - public T child(Integer index, Double timeout, Boolean eleOnly) { - String loc = eleOnly ? "*" : "node()"; - List ts = this._ele("xpath./" + loc, timeout, index, true, false, null); - if (!ts.isEmpty()) { - return ts.get(0); - } else if (Settings.raiseWhenEleNotFound) { - throw new ElementNotFoundError("child()", Map.of("loc", "", "index", index, "eleOnly", eleOnly)); - } - return null; - } - - /** - * 返回直接子元素元素或节点组成的列表,可用查询语法筛选 - * - * @return 直接子元素或节点文本组成的列表 - */ - public T child() { - return child(""); - } - - /** - * 返回直接子元素元素或节点组成的列表,可用查询语法筛选 - * - * @param loc 查询语句 - * @return 直接子元素或节点文本组成的列表 - */ - public T child(String loc) { - return child(loc.isEmpty() ? null : loc, 1); - } - - /** - * 返回直接子元素元素或节点组成的列表,可用查询语法筛选 - * - * @param loc 查询语句 - * @param index 第几个查询结果,1开始 - * @return 直接子元素或节点文本组成的列表 - */ - public T child(String loc, Integer index) { - return child(loc, index, null); - } - - /** - * 返回直接子元素元素或节点组成的列表,可用查询语法筛选 - * - * @param loc 查询语句 - * @param index 第几个查询结果,1开始 - * @param timeout 查找节点的超时时间 - * @return 直接子元素或节点文本组成的列表 - */ - public T child(String loc, Integer index, Double timeout) { - return child(loc, index, timeout, true); - } - - /** - * 返回直接子元素元素或节点组成的列表,可用查询语法筛选 - * - * @param loc 查询语句 - * @param index 第几个查询结果,1开始 - * @param timeout 查找节点的超时时间 - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 直接子元素或节点文本组成的列表 - */ - public T child(String loc, Integer index, Double timeout, Boolean eleOnly) { - if (loc == null || loc.isEmpty()) { - loc = eleOnly ? "*" : "node()"; - } else { - By by = Locator.getLoc(loc, true, false); - if (by.getName().equals(BySelect.CSS_SELECTOR)) - throw new IllegalArgumentException("此css selector语法不受支持,请换成xpath。"); - else loc = by.getValue().replaceAll("^[.\\s/]+", ""); - } - - List ts = this._ele("xpath./" + loc, timeout, index, true, false, null); - if (!ts.isEmpty()) { - return ts.get(0); - } else if (Settings.raiseWhenEleNotFound) { - throw new ElementNotFoundError("child()", Map.of("loc", loc, "index", index, "eleOnly", eleOnly)); - } - return null; - } - - /** - * 返回直接子元素元素或节点组成的列表,可用查询语法筛选 - * - * @param by 查询语句 - * @return 直接子元素或节点文本组成的列表 - */ - public T child(By by) { - return child(by, 1); - } - - /** - * 返回直接子元素元素或节点组成的列表,可用查询语法筛选 - * - * @param by 查询语句 - * @param index 第几个查询结果,1开始 - * @return 直接子元素或节点文本组成的列表 - */ - public T child(By by, Integer index) { - return child(by, index, null); - } - - /** - * 返回直接子元素元素或节点组成的列表,可用查询语法筛选 - * - * @param by 查询语句 - * @param index 第几个查询结果,1开始 - * @param timeout 查找节点的超时时间 - * @return 直接子元素或节点文本组成的列表 - */ - public T child(By by, Integer index, Double timeout) { - return child(by, index, timeout, true); - } - - /** - * 返回直接子元素元素或节点组成的列表,可用查询语法筛选 - * - * @param by 查询语句 - * @param index 第几个查询结果,1开始 - * @param timeout 查找节点的超时时间 - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 直接子元素或节点文本组成的列表 - */ - public T child(By by, Integer index, Double timeout, Boolean eleOnly) { - String loc; - if (by == null) { - loc = eleOnly ? "*" : "node()"; - } else { - by = Locator.getLoc(by, true, false); - if (by.getName().equals(BySelect.CSS_SELECTOR)) - throw new IllegalArgumentException("此css selector语法不受支持,请换成xpath。"); - else loc = by.getValue().replaceAll("^[.\\s/]+", ""); - } - - List ts = this._ele("xpath./" + loc, timeout, index, true, false, null); - if (!ts.isEmpty()) { - return ts.get(0); - } else if (Settings.raiseWhenEleNotFound) { - throw new ElementNotFoundError("child()", Map.of("loc", loc, "index", index, "eleOnly", eleOnly)); - } - return null; - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @return 兄弟元素 - */ - public T prev() { - return prev(""); - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param loc 用于筛选的查询语法 - * @return 兄弟元素 - */ - public T prev(String loc) { - return prev(loc.isEmpty() ? null : loc, null); - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param loc 用于筛选的查询语法 - * @param index 前面第几个查询结果,0开始 - * @return 兄弟元素 - */ - - public T prev(String loc, Integer index) { - return prev(loc, index, null); - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param loc 用于筛选的查询语法 - * @param index 前面第几个查询结果,0开始 - * @param timeout 查找节点的超时时间(秒) - * @return 兄弟元素 - */ - public T prev(String loc, Integer index, Double timeout) { - return prev(loc, index, timeout, true); - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param loc 用于筛选的查询语法 - * @param index 前面第几个查询结果,1开始 - * @param timeout 查找节点的超时时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 兄弟元素 - */ - public T prev(String loc, Integer index, Double timeout, Boolean eleOnly) { - return this.getRelative("prev()", "preceding", true, loc, index, timeout, eleOnly); - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param by 用于筛选的元素 - * @return 兄弟元素 - */ - public T prev(By by) { - return prev(by, 1); - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param by 用于筛选的元素 - * @param index 前面第几个查询结果,0开始 - * @return 兄弟元素 - */ - public T prev(By by, Integer index) { - return prev(by, index, null); - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param by 用于筛选的元素 - * @param index 前面第几个查询结果,0开始 - * @param timeout 查找节点的超时时间(秒) - * @return 兄弟元素 - */ - public T prev(By by, Integer index, Double timeout) { - return prev(by, index, timeout, true); - } - - public T prev(By by, Integer index, Double timeout, Boolean eleOnly) { - return this.getRelative("prev()", "preceding", true, by, index, timeout, eleOnly); - } - - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param index 前面第几个查询结果,0开始 - * @return 兄弟元素 - */ - public T prev(Integer index) { - return prev(index, null); - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param index 前面第几个查询结果,0开始 - * @param timeout 查找节点的超时时间(秒) - * @return 兄弟元素 - */ - public T prev(Integer index, Double timeout) { - return prev(index, timeout, true); - } - - public T prev(Integer index, Double timeout, Boolean eleOnly) { - return this.getRelative("prev()", "preceding", true, index, timeout, eleOnly); - } - - /** - * 返回后面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param by 用于筛选的查询语法 - * @param index 第几个查询结果,1开始 - * @param timeout 查找节点的超时时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 兄弟元素 - */ - @Override - public T next(By by, Integer index, Double timeout, Boolean eleOnly) { - return this.getRelative("next()", "following", true, by, index, timeout, eleOnly); - } - - /** - * 返回后面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param loc 用于筛选的查询语法 - * @param index 第几个查询结果,1开始 - * @param timeout 查找节点的超时时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 兄弟元素 - */ - - @Override - public T next(String loc, Integer index, Double timeout, Boolean eleOnly) { - return this.getRelative("next()", "following", true, loc, index, timeout, eleOnly); - } - - /** - * 返回当前元素后面一个符合条件的同级元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param index 第几个查询结果,0开始 - * @param timeout 查找节点的超时时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 兄弟元素或节点文本 - */ - @Override - public T next(Integer index, Double timeout, Boolean eleOnly) { - return this.getRelative("next()", "following", true, index, timeout, eleOnly); - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param by 用于筛选的查询语法 - * @return 本元素前面的某个元素或节点 - */ - public T before(By by) { - return before(by, null); - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param by 用于筛选的查询语法 - * @param index 前面第几个查询结果,1开始 - * @return 本元素前面的某个元素或节点 - */ - public T before(By by, Integer index) { - return before(by, index, null); - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param by 用于筛选的查询语法 - * @param index 前面第几个查询结果,1开始 - * @param timeout 查找节点的超时时间(秒) - * @return 本元素前面的某个元素或节点 - */ - public T before(By by, Integer index, Double timeout) { - return before(by, index, timeout, true); - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param by 用于筛选的查询语法 - * @param index 前面第几个查询结果,1开始 - * @param timeout 查找节点的超时时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 本元素前面的某个元素或节点 - */ - public T before(By by, Integer index, Double timeout, Boolean eleOnly) { - return this.getRelative("before()", "preceding", false, by, index, timeout, eleOnly); - } - - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @return 本元素前面的某个元素或节点 - */ - public T before() { - return before(""); - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param loc 用于筛选的查询语法 - * @return 本元素前面的某个元素或节点 - */ - public T before(String loc) { - return before(loc.isEmpty() ? null : loc, null); - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param loc 用于筛选的查询语法 - * @param index 前面第几个查询结果,1开始 - * @return 本元素前面的某个元素或节点 - */ - public T before(String loc, Integer index) { - return before(loc, index, null); - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param loc 用于筛选的查询语法 - * @param index 前面第几个查询结果,1开始 - * @param timeout 查找节点的超时时间(秒) - * @return 本元素前面的某个元素或节点 - */ - public T before(String loc, Integer index, Double timeout) { - return before(loc, index, timeout, true); - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param loc 用于筛选的查询语法 - * @param index 前面第几个查询结果,1开始 - * @param timeout 查找节点的超时时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 本元素前面的某个元素或节点 - */ - public T before(String loc, Integer index, Double timeout, Boolean eleOnly) { - return this.getRelative("before()", "preceding", false, loc, index, timeout, eleOnly); - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param index 前面第几个查询结果,1开始 - * @return 本元素前面的某个元素或节点 - */ - public T before(Integer index) { - return before(index, null); - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param index 前面第几个查询结果,1开始 - * @param timeout 查找节点的超时时间(秒) - * @return 本元素前面的某个元素或节点 - */ - public T before(Integer index, Double timeout) { - return before(index, timeout, true); - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param index 前面第几个查询结果,1开始 - * @param timeout 查找节点的超时时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 本元素前面的某个元素或节点 - */ - public T before(Integer index, Double timeout, Boolean eleOnly) { - return this.getRelative("before()", "preceding", false, index, timeout, eleOnly); - } - - /** - * 返回后面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param by 用于筛选的查询语法 - * @return 本元素后面的某个元素或节点 - */ - public T after(By by) { - return after(by, null); - } - - /** - * 返回后面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param by 用于筛选的查询语法 - * @param index 前面第几个查询结果,1开始 - * @return 本元素后面的某个元素或节点 - */ - public T after(By by, Integer index) { - return after(by, index, null); - } - - /** - * 返回后面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param by 用于筛选的查询语法 - * @param index 前面第几个查询结果,1开始 - * @param timeout 查找节点的超时时间(秒) - * @return 本元素后面的某个元素或节点 - */ - public T after(By by, Integer index, Double timeout) { - return after(by, index, timeout, true); - } - - /** - * 返回后面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param by 用于筛选的查询语法 - * @param index 前面第几个查询结果,1开始 - * @param timeout 查找节点的超时时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 本元素后面的某个元素或节点 - */ - public T after(By by, Integer index, Double timeout, Boolean eleOnly) { - return this.getRelative("after()", "following", false, by, index, timeout, eleOnly); - } - - /** - * 返回后面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @return 本元素后面的某个元素或节点 - */ - public T after() { - return after(""); - } - - /** - * 返回后面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param loc 用于筛选的查询语法 - * @return 本元素后面的某个元素或节点 - */ - public T after(String loc) { - return after(loc.isEmpty() ? null : loc, null); - } - - /** - * 返回后面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param loc 用于筛选的查询语法 - * @param index 前面第几个查询结果,1开始 - * @return 本元素后面的某个元素或节点 - */ - public T after(String loc, Integer index) { - return after(loc, index, null); - } - - /** - * 返回后面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param loc 用于筛选的查询语法 - * @param index 前面第几个查询结果,1开始 - * @param timeout 查找节点的超时时间(秒) - * @return 本元素后面的某个元素或节点 - */ - public T after(String loc, Integer index, Double timeout) { - return after(loc, index, timeout, true); - } - - /** - * 返回后面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param loc 用于筛选的查询语法 - * @param index 前面第几个查询结果,1开始 - * @param timeout 查找节点的超时时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 本元素后面的某个元素或节点 - */ - public T after(String loc, Integer index, Double timeout, Boolean eleOnly) { - return this.getRelative("after()", "following", false, loc, index, timeout, eleOnly); - } - - /** - * 返回后面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param index 前面第几个查询结果,1开始 - * @return 本元素后面的某个元素或节点 - */ - public T after(Integer index) { - return after(index, null); - } - - /** - * 返回后面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param index 前面第几个查询结果,1开始 - * @param timeout 查找节点的超时时间(秒) - * @return 本元素后面的某个元素或节点 - */ - public T after(Integer index, Double timeout) { - return after(index, timeout, true); - } - - /** - * 返回后面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param index 前面第几个查询结果,1开始 - * @param timeout 查找节点的超时时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 本元素后面的某个元素或节点 - */ - public T after(Integer index, Double timeout, Boolean eleOnly) { - return this.getRelative("after()", "following", false, index, timeout, eleOnly); - } - - /** - * 返回直接子元素元素或节点组成的列表,可用查询语法筛选 - * - * @param by 用于筛选的查询语法 - * @return 本元素后面的某个元素或节点 - */ - public List children(By by) { - return children(by, null); - } - - /** - * 返回直接子元素元素或节点组成的列表,可用查询语法筛选 - * - * @param by 用于筛选的查询语法 - * @param timeout 查找节点的超时时间(秒) - * @return 本元素后面的某个元素或节点 - */ - public List children(By by, Double timeout) { - return children(by, timeout, true); - } - - /** - * 返回直接子元素元素或节点组成的列表,可用查询语法筛选 - * - * @param by 用于筛选的查询语法 - * @param timeout 查找节点的超时时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 本元素后面的某个元素或节点 - */ - public List children(By by, Double timeout, Boolean eleOnly) { - String loc; - if (by == null) { - loc = eleOnly ? "*" : "node()"; - } else { - by = Locator.getLoc(by, true, false); - if (by.getName().equals(BySelect.CSS_SELECTOR)) - throw new IllegalArgumentException("此css selector语法不受支持,请换成xpath。"); - else loc = by.getValue().replaceAll("^[.\\s/]+", ""); - } - loc = "xpath:./" + loc; - return this._ele(loc, timeout, null, true, null, null); - } - - /** - * 返回直接子元素元素或节点组成的列表,可用查询语法筛选 - * - * @return 本元素后面的某个元素或节点 - */ - public List children() { - return children(""); - } - - /** - * 返回直接子元素元素或节点组成的列表,可用查询语法筛选 - * - * @param loc 用于筛选的查询语法 - * @return 本元素后面的某个元素或节点 - */ - public List children(String loc) { - return children(loc.isEmpty() ? null : loc, null); - } - - /** - * 返回直接子元素元素或节点组成的列表,可用查询语法筛选 - * - * @param loc 用于筛选的查询语法 - * @param timeout 查找节点的超时时间(秒) - * @return 本元素后面的某个元素或节点 - */ - public List children(String loc, Double timeout) { - return children(loc, timeout, true); - } - - /** - * 返回直接子元素元素或节点组成的列表,可用查询语法筛选 - * - * @param loc 用于筛选的查询语法 - * @param timeout 查找节点的超时时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 本元素后面的某个元素或节点 - */ - public List children(String loc, Double timeout, Boolean eleOnly) { - if (loc == null || loc.isEmpty()) { - loc = eleOnly ? "*" : "node()"; - } else { - By by = Locator.getLoc(loc, true, false); - if (by.getName().equals(BySelect.CSS_SELECTOR)) - throw new IllegalArgumentException("此css selector语法不受支持,请换成xpath。"); - else { - loc = by.getValue().replaceAll("^[.\\s/]+", ""); - } - } - loc = "xpath:./" + loc; - return this._ele(loc, timeout, null, true, null, null); - } - - - /** - * 获取前面符合条件的同级列表 - * - * @return 同级元素或节点文本组成的列表 - */ - public List prevs() { - return prevs("", null); - } - - /** - * 获取前面符合条件的同级列表 - * - * @param loc 查询元素 - * @return 同级元素或节点文本组成的列表 - */ - public List prevs(String loc) { - return prevs(loc.isEmpty() ? null : loc, null); - } - - /** - * 获取前面符合条件的同级列表 - * - * @param loc 查询元素 - * @param timeout 等待时间 - * @return 同级元素或节点文本组成的列表 - */ - public List prevs(String loc, Double timeout) { - return prevs(loc, timeout, true); - } - - /** - * 回前面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param loc 查询元素 - * @param timeout 等待时间 - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 兄弟元素的列表 - */ - public List prevs(String loc, Double timeout, Boolean eleOnly) { - return this.getRelatives(null, loc, "preceding", true, timeout, eleOnly); - } - - /** - * 获取前面符合条件的同级列表 - * - * @param by 查询元素 - * @return 同级元素或节点文本组成的列表 - */ - public List prevs(By by) { - return prevs(by, null); - } - - /** - * 获取前面符合条件的同级列表 - * - * @param by 查询元素 - * @param timeout 等待时间 - * @return 同级元素或节点文本组成的列表 - */ - public List prevs(By by, Double timeout) { - return prevs(by, timeout, true); - } - - - /** - * 回前面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param by 查询元素 - * @param timeout 等待时间 - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 兄弟元素的列表 - */ - public List prevs(By by, Double timeout, Boolean eleOnly) { - return this.getRelatives(null, by, "preceding", true, timeout, eleOnly); - } - - /** - * 返回后面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param by 用于筛选的查询语法 - * @return 兄弟元素或节点文本组成的列表 - */ - public List nexts(By by) { - return nexts(by, null); - } - - /** - * 返回后面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param by 用于筛选的查询语法 - * @param timeout 查找节点的超时时间 - * @return 兄弟元素或节点文本组成的列表 - */ - public List nexts(By by, Double timeout) { - return nexts(by, timeout, true); - } - - /** - * 返回后面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param by 用于筛选的查询语法 - * @param timeout 查找节点的超时时间 - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 兄弟元素或节点文本组成的列表 - */ - public List nexts(By by, Double timeout, Boolean eleOnly) { - return this.getRelatives(null, by, "following", true, timeout, eleOnly); - } - - /** - * 返回后面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @return 兄弟元素或节点文本组成的列表 - */ - - public List nexts() { - return nexts(""); - } - - /** - * 返回后面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param loc 用于筛选的查询语法 - * @return 兄弟元素或节点文本组成的列表 - */ - public List nexts(String loc) { - return nexts(loc.isEmpty() ? null : loc, null); - } - - /** - * 返回后面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param loc 用于筛选的查询语法 - * @param timeout 查找节点的超时时间 - * @return 兄弟元素或节点文本组成的列表 - */ - public List nexts(String loc, Double timeout) { - return nexts(loc, timeout, true); - } - - /** - * 返回后面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param loc 用于筛选的查询语法 - * @param timeout 查找节点的超时时间 - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 兄弟元素或节点文本组成的列表 - */ - public List nexts(String loc, Double timeout, Boolean eleOnly) { - return this.getRelatives(null, loc, "following", true, timeout, eleOnly); - } - - /** - * 返回后面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param by 用于筛选的查询语法 - * @return 本元素前面的元素或节点组成的列表 - */ - public List befores(By by) { - return this.befores(by, null); - } - - /** - * 返回后面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param by 用于筛选的查询语法 - * @param timeout 查找节点的超时时间(秒) - * @return 本元素前面的元素或节点组成的列表 - */ - public List befores(By by, Double timeout) { - return this.befores(by, timeout, true); - } - - /** - * 返回后面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param by 用于筛选的查询语法 - * @param timeout 查找节点的超时时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 本元素前面的元素或节点组成的列表 - */ - public List befores(By by, Double timeout, Boolean eleOnly) { - return this.getRelatives(null, by, "preceding", false, timeout, eleOnly); - } - - /** - * 返回后面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @return 本元素前面的元素或节点组成的列表 - */ - public List befores() { - return this.befores(""); - } - - /** - * 返回后面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param loc 用于筛选的查询语法 - * @return 本元素前面的元素或节点组成的列表 - */ - public List befores(String loc) { - return this.befores(loc.isEmpty() ? null : loc, null); - } - - /** - * 返回后面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param loc 用于筛选的查询语法 - * @param timeout 查找节点的超时时间(秒) - * @return 本元素前面的元素或节点组成的列表 - */ - public List befores(String loc, Double timeout) { - return this.befores(loc, timeout, true); - } - - /** - * 返回后面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param loc 用于筛选的查询语法 - * @param timeout 查找节点的超时时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 本元素前面的元素或节点组成的列表 - */ - public List befores(String loc, Double timeout, Boolean eleOnly) { - return this.getRelatives(null, loc, "preceding", false, timeout, eleOnly); - } - - /** - * 返回前面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param by 用于筛选的查询语法 - * @return 本元素前面的元素或节点组成的列表 - */ - public List afters(By by) { - return this.afters(by, null); - } - - - /** - * 返回前面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param by 用于筛选的查询语法 - * @param timeout 查找节点的超时时间(秒) - * @return 本元素前面的元素或节点组成的列表 - */ - public List afters(By by, Double timeout) { - return this.afters(by, timeout, true); - } - - /** - * 返回前面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param by 用于筛选的查询语法 - * @param timeout 查找节点的超时时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 本元素前面的元素或节点组成的列表 - */ - public List afters(By by, Double timeout, Boolean eleOnly) { - return this.getRelatives(null, by, "following", false, timeout, eleOnly); - } - - - /** - * 返回前面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @return 本元素前面的元素或节点组成的列表 - */ - public List afters() { - return this.afters(""); - } - - /** - * 返回前面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param loc 用于筛选的查询语法 - * @return 本元素前面的元素或节点组成的列表 - */ - public List afters(String loc) { - return this.afters(loc.isEmpty() ? null : loc, null); - } - - /** - * 返回前面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param loc 用于筛选的查询语法 - * @param timeout 查找节点的超时时间(秒) - * @return 本元素前面的元素或节点组成的列表 - */ - public List afters(String loc, Double timeout) { - return this.afters(loc, timeout, true); - } - - /** - * 返回前面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param loc 用于筛选的查询语法 - * @param timeout 查找节点的超时时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 本元素前面的元素或节点组成的列表 - */ - public List afters(String loc, Double timeout, Boolean eleOnly) { - return this.getRelatives(null, loc, "following", false, timeout, eleOnly); - } - - /** - * *获取一个亲戚元素或节点,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param func: 方法名称 - * @param direction: 方向,'following' 或 'preceding' - * @param brother 查找范围,在同级查找还是整个dom前后查找 - * @param loc: 用于筛选的查询语法 - * @param timeout: 查找节点的超时时间(秒) - * @param eleOnly: 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 本元素前面的某个元素或节点 - */ - private T getRelative(String func, String direction, Boolean brother, Integer loc, Double timeout, Boolean eleOnly) { - List relatives = this.getRelatives(loc, "", direction, brother, timeout, eleOnly); - if (!relatives.isEmpty()) return relatives.get(0); - if (Settings.raiseWhenEleNotFound) - throw new ElementNotFoundError(func, Map.of("loc", "", "index", loc, "eleOnly", eleOnly)); - return null; - } - - /** - * *获取一个亲戚元素或节点,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param func: 方法名称 - * @param direction: 方向,'following' 或 'preceding' - * @param brother 查找范围,在同级查找还是整个dom前后查找 - * @param loc: 用于筛选的查询语法 - * @param index: 前面第几个查询结果,1开始 - * @param timeout: 查找节点的超时时间(秒) - * @param eleOnly: 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 本元素前面的某个元素或节点 - */ - private T getRelative(String func, String direction, Boolean brother, String loc, Integer index, Double timeout, Boolean eleOnly) { - List relatives = this.getRelatives(index, loc, direction, brother, timeout, eleOnly); - if (!relatives.isEmpty()) return relatives.get(0); - if (Settings.raiseWhenEleNotFound) - throw new ElementNotFoundError(func, Map.of("loc", "", "index", index, "eleOnly", eleOnly)); - return null; - } - - /** - * *获取一个亲戚元素或节点,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param func: 方法名称 - * @param direction: 方向,'following' 或 'preceding' - * @param brother 查找范围,在同级查找还是整个dom前后查找 - * @param by: 用于筛选的查询语法 - * @param index: 前面第几个查询结果,1开始 - * @param timeout: 查找节点的超时时间(秒) - * @param eleOnly: 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 本元素前面的某个元素或节点 - */ - private T getRelative(String func, String direction, Boolean brother, By by, Integer index, Double timeout, Boolean eleOnly) { - List relatives = this.getRelatives(index, by, direction, brother, timeout, eleOnly); - if (!relatives.isEmpty()) return relatives.get(0); - if (Settings.raiseWhenEleNotFound) - throw new ElementNotFoundError(func, Map.of("loc", "", "index", index, "eleOnly", eleOnly)); - return null; - } - - /** - * @param index 获取第几个,该参数不为null时只获取该编号的元素 - * @param loc 用于筛选的查询语法 - * @param direction 'following' 或 'preceding',查找的方向 - * @param brother 查找范围,在同级查找还是整个dom前后查找 - * @param timeout 查找等待时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 元素对象或字符串 - */ - private List getRelatives(Integer index, String loc, String direction, Boolean brother, Double timeout, Boolean eleOnly) { - //获取查到范围 - String brotherStr = brother ? "-sibling" : ""; - if (loc == null || loc.isEmpty()) { - loc = eleOnly ? "*" : "node()"; - } else { - By by = Locator.getLoc(loc, true, false); - if (!by.getName().equals(BySelect.CSS_SELECTOR)) - throw new IllegalArgumentException("此css selector语法不受支持,请换成xpath。"); - else loc = by.getValue().replaceAll("^[.\\s/]+", ""); - } - loc = "xpath:./" + direction + brotherStr + "::" + loc; - if (index != null) index = "following".equals(direction) ? index : -index; - return this._ele(loc, timeout, index, true, false, null); - } - - /** - * @param index 获取第几个,该参数不为null时只获取该编号的元素 - * @param by 用于筛选的查询语法 - * @param direction 'following' 或 'preceding',查找的方向 - * @param brother 查找范围,在同级查找还是整个dom前后查找 - * @param timeout 查找等待时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 元素对象或字符串 - */ - private List getRelatives(Integer index, By by, String direction, Boolean brother, Double timeout, Boolean eleOnly) { - //获取查到范围 - String brotherStr = brother ? "-sibling" : ""; - String loc; - if (by == null) { - loc = eleOnly ? "*" : "node()"; - } else { - by = Locator.getLoc(by, true, false); - if (!by.getName().equals(BySelect.CSS_SELECTOR)) - throw new IllegalArgumentException("此css selector语法不受支持,请换成xpath。"); - else loc = by.getValue().replaceAll("^[.\\s/]+", ""); - } - loc = "xpath:./" + direction + brotherStr + "::" + loc; - if (index != null) index = "following".equals(direction) ? index : -index; - return this._ele(loc, timeout, index, true, false, null); - } - - //---------------------------------------------- - - /** - * @return 返回元素所有属性及值 - */ - public abstract Map attrs(); - - /** - * 返回处理的元素内文本 - */ - public abstract String text(); - - /** - * 返回未格式化处理的元素内文本 - */ - public abstract String rawText(); - - /** - * 返回attribute属性值 - * - * @param attr 属性名 - * @return 属性值文本,没有该属性返回null - */ - public abstract String attr(String attr); - - /** - * 获取css路径或xpath路径 - * - * @param mode 'css' 或 'xpath' - * @return css路径或xpath路径 - */ - protected abstract String getElePath(ElePathMode mode); -} diff --git a/java/src/main/java/com/ll/DrissonPage/base/Driver.java b/java/src/main/java/com/ll/DrissonPage/base/Driver.java deleted file mode 100644 index fc53df3..0000000 --- a/java/src/main/java/com/ll/DrissonPage/base/Driver.java +++ /dev/null @@ -1,450 +0,0 @@ -package com.ll.DrissonPage.base; - -import com.alibaba.fastjson.JSON; -import com.alibaba.fastjson.JSONObject; -import com.ll.DrissonPage.functions.Settings; -import lombok.Getter; -import lombok.Setter; -import okhttp3.*; -import okio.ByteString; -import org.jetbrains.annotations.NotNull; - -import java.util.ArrayList; -import java.util.HashMap; -import java.util.Map; -import java.util.concurrent.BlockingQueue; -import java.util.concurrent.ConcurrentHashMap; -import java.util.concurrent.LinkedBlockingQueue; -import java.util.concurrent.TimeUnit; -import java.util.concurrent.atomic.AtomicBoolean; -import java.util.concurrent.atomic.AtomicInteger; -import java.util.concurrent.atomic.AtomicReference; - -/** - * 驱动 okHTTP框架 - * - * @author 陆 - * @address click - * @original DrissionPage - */ - -public class Driver { - /** - * 标签id - */ - @Getter - private final String id; - /** - * 浏览器连接地址 - */ - @Getter - private final String address; - /** - * 标签页类型 - */ - @Getter - private final String type; - @Setter - @Getter - private boolean debug; - private final String websocketUrl; - private final AtomicInteger curId; - /** - * 会话返回值 - */ - private final AtomicReference webSocketMsg = new AtomicReference<>(null); - private final Thread recvThread; - private final Thread handleEventThread; - @Getter - private AtomicBoolean stopped; - private final BlockingQueue> eventQueue; - private final BlockingQueue> immediateEventQueue; - private final ConcurrentHashMap eventHandlers; - private final ConcurrentHashMap immediateEventHandlers; - private final ConcurrentHashMap>> methodResults; - /** - * 创建这个驱动的对象 - */ - @Getter - @Setter - private Occupant occupant; - private boolean alertFlag; - private static final OkHttpClient okHttpClient = new OkHttpClient().newBuilder() - .connectTimeout(5, TimeUnit.SECONDS) - .callTimeout(5, TimeUnit.SECONDS) - .readTimeout(5, TimeUnit.SECONDS) - .pingInterval(5, TimeUnit.SECONDS).build(); - /** - * 会话驱动 - */ - private WebSocket ws; - private Thread handleImmediateEventThread; - - public Driver(String tabId, String tabType, String address) { - this(tabId, tabType, address, null); - } - - /** - * 驱动 - * - * @param tabId 标签id - * @param tabType 标签页类型 - * @param address 浏览器连接地址 - */ - public Driver(String tabId, String tabType, String address, Occupant occupant) { - this.id = tabId; - this.address = address; - this.type = tabType; - this.occupant = occupant; - this.debug = false; - this.alertFlag = false; - this.websocketUrl = "ws://" + address + "/devtools/" + tabType + "/" + tabId; - this.curId = new AtomicInteger(0); - this.ws = null; - - - this.recvThread = new Thread(this::recvLoop); - this.handleEventThread = new Thread(this::handleEventLoop); - this.recvThread.setDaemon(true); - this.handleEventThread.setDaemon(true); - this.handleImmediateEventThread = null; - - this.stopped = new AtomicBoolean(); - - this.eventHandlers = new ConcurrentHashMap<>(); - this.immediateEventHandlers = new ConcurrentHashMap<>(); - this.methodResults = new ConcurrentHashMap<>(); - this.eventQueue = new LinkedBlockingQueue<>(); - this.immediateEventQueue = new LinkedBlockingQueue<>(); - start(); - } - - /** - * 发送信息到浏览器,并返回浏览器返回的信息 - * - * @param message 发送给浏览器的数据 - * @param timeout 超时时间,为null表示无时间 - * @return 浏览器返回的数据 - */ - private JSONObject send(Map message, double timeout) { - message = new HashMap<>(message); - int wsId = curId.incrementAndGet(); - message.put("id", wsId); - String messageJson = JSON.toJSONString(message); - - if (this.debug) System.out.println("发->" + messageJson); - //计算等待时间 - long endTime = (long) (System.currentTimeMillis() + timeout * 1000L); - LinkedBlockingQueue> value = new LinkedBlockingQueue<>(); - methodResults.put(wsId, value); - try { - ws.send(messageJson); - if (timeout == 0) { - methodResults.remove(wsId); - return new JSONObject(Map.of("id", wsId, "result", Map.of())); - } - } catch (Exception e) { - e.printStackTrace(); - methodResults.remove(wsId); - return new JSONObject(Map.of("error", Map.of("message", "connection disconnected"), "type", "connection_error")); - } - int i = 5; - long endTimes = System.currentTimeMillis() + 1000L; - while (!stopped.get()) { - try { - Map result = methodResults.get(wsId).poll(5, TimeUnit.SECONDS); - if (result == null && System.currentTimeMillis() < endTimes) continue; - if (result == null && i > 0 && System.currentTimeMillis() > endTimes) { - i--; - endTimes = System.currentTimeMillis() + 1000L; - if (debug) System.out.println("超时或者丢包,重新发送:->" + messageJson); - ws.send(messageJson); - continue; - } - methodResults.remove(wsId); - if (result == null) throw new NullPointerException(); - return new JSONObject(result); - } catch (InterruptedException | NullPointerException | IllegalArgumentException e) { -// e.printStackTrace(); - String string = message.get("method").toString(); - if (alertFlag && string.startsWith("Input.") || string.startsWith("Runtime.")) { - return new JSONObject(Map.of("error", Map.of("message", "alert exists."), "type", "alert_exists")); - } - if (timeout > 0 && System.currentTimeMillis() > endTime) { - methodResults.remove(wsId); - return alertFlag ? new JSONObject(Map.of("error", Map.of("message", "alert exists."), "type", "alert_exists")) : new JSONObject(Map.of("error", Map.of("message", "timeout"), "type", "timeout")); - } - } - } - - return new JSONObject(Map.of("error", Map.of("message", "connection disconnected"), "type", "connection_error")); - } - - /** - * 接收浏览器信息的守护线程方法 - */ - private void recvLoop() { - while (!stopped.get()) { - JSONObject msg; - try { - String andSet = webSocketMsg.getAndSet(null); - if (andSet != null) { - msg = JSONObject.parseObject(andSet); - } else continue; - } catch (Exception e) { - if (stop()) return; - return; - - } - if (this.debug) System.out.println("<-收" + msg); - - if (msg.containsKey("method")) { - if (msg.getString("method").startsWith("Page.javascriptDialog")) { - alertFlag = msg.getString("method").endsWith("Opening"); - } - MyRunnable function = immediateEventHandlers.get(msg.getString("method")); - if (function != null) { - this.handleImmediateEvent(function, msg.getOrDefault("params", new HashMap<>())); - } else { - try { - eventQueue.put(msg); - } catch (InterruptedException e) { - throw new RuntimeException(e); - } - } - } else { - int i = 1000; - Integer integer = msg.getInteger("id"); - while (i-- > 0 && integer != null && !methodResults.containsKey(integer)) { - try { - Thread.sleep(10); - } catch (InterruptedException e) { - throw new RuntimeException(e); - } - } - if (methodResults.containsKey(integer)) { - try { - methodResults.get(integer).put(msg); - } catch (InterruptedException e) { - throw new RuntimeException(e); - } - } else if (this.debug) { - System.out.println("未知错误->" + msg); - - } - } - - } - } - - /** - * 当接收到浏览器信息,执行已绑定的方法 - */ - private void handleEventLoop() { - while (!stopped.get()) { - Map event; - try { - event = eventQueue.poll(2, TimeUnit.SECONDS); - } catch (InterruptedException e) { - continue; - } - - if (event != null) { - MyRunnable function = eventHandlers.get(event.get("method").toString()); - if (function != null) { - function.setMessage(event.get("params")); - function.run(); - } - } - this.eventQueue.poll(); - - } - } - - private void handleImmediateEventLoop() { - while (!stopped.get() && !immediateEventQueue.isEmpty()) { - Map event; - try { - event = immediateEventQueue.poll(2, TimeUnit.SECONDS); - } catch (InterruptedException e) { - e.printStackTrace(); - continue; - } - if (event != null) { - MyRunnable function = immediateEventHandlers.get(event.get("method").toString()); - if (function != null) { - function.setMessage(event.get("params")); - function.run(); - } - } - - } - } - - /** - * 处理立即执行的动作 - * - * @param function 要运行下方法 - * @param params 方法参数 - */ - private void handleImmediateEvent(MyRunnable function, Object params) { - Map func = new HashMap<>(); - func.put("method", function); - func.put("params", params); - immediateEventQueue.add(func); - - if (handleImmediateEventThread == null || !handleImmediateEventThread.isAlive()) { - handleImmediateEventThread = new Thread(this::handleImmediateEventLoop); - handleImmediateEventThread.setDaemon(true); - handleImmediateEventThread.start(); - } - } - - /** - * 执行cdp方法 - * - * @param method 方法 - * @return 执行结果 - */ - public Object run(String method) { - return run(method, new HashMap<>()); - } - - /** - * 执行cdp方法 - * - * @param method 方法 - * @param params 参数 - * @return 执行结果 - */ - public Object run(String method, Map params) { - if (stopped.get()) return Map.of("error", "connection disconnected", "type", "connection_error"); - params = new HashMap<>(params); - Object timeout1 = params.remove("_timeout"); - double timeout = timeout1 != null ? Float.parseFloat(timeout1.toString()) : Settings.cdpTimeout; - - JSONObject result = this.send(Map.of("method", method, "params", params), timeout); - if (!result.containsKey("result") && result.containsKey("error")) { - HashMap map = new HashMap<>(); - map.put("error", result.getJSONObject("error").get("message")); - map.put("type", result.getOrDefault("type", "call_method_error")); - map.put("method", method); - map.put("args", params); - map.put("timeout", timeout); - return JSON.toJSONString(map); - } else { - return JSON.toJSONString(result.get("result")); - } - } - - - /** - * 启动连接 - */ - private void start() { - this.stopped.set(false); - try { - Request build = new Request(new HttpUrl("socket", "", "", "", 80, new ArrayList<>(), null, null, this.websocketUrl), "GET", Headers.of(), null, new HashMap<>()).newBuilder().url(this.websocketUrl).build(); - ws = okHttpClient.newWebSocket(build, new WebSocketListener() { - @Override - public void onClosed(@NotNull WebSocket webSocket, int code, @NotNull String reason) { - // 关闭事件处理 - stop(); - super.onClosed(webSocket, code, reason); - } - - @Override - public void onClosing(@NotNull WebSocket webSocket, int code, @NotNull String reason) { - super.onClosing(webSocket, code, reason); - } - - @Override - public void onFailure(@NotNull WebSocket webSocket, @NotNull Throwable t, Response response) { - super.onFailure(webSocket, t, response); - } - - @Override - public void onMessage(@NotNull WebSocket webSocket, @NotNull String text) { - webSocketMsg.set(text); - super.onMessage(webSocket, text); - } - - @Override - public void onMessage(@NotNull WebSocket webSocket, @NotNull ByteString bytes) { - super.onMessage(webSocket, bytes); - } - - @Override - public void onOpen(@NotNull WebSocket webSocket, @NotNull Response response) { - super.onOpen(webSocket, response); - } - - }); - recvThread.start(); - handleEventThread.start(); - } catch (Exception e) { - e.printStackTrace(); - stop(); - } - } - - /** - * 中断连接 - */ - public boolean stop() { - stop1(); - while (this.recvThread.isAlive() || this.handleEventThread.isAlive()) { - try { - Thread.sleep(100); - } catch (InterruptedException e) { - throw new RuntimeException(e); - } - } - return true; - } - - /** - * 中断连接 - */ - private void stop1() { - if (debug) System.out.println("关闭"); - if (stopped.get()) return; - stopped.set(!stopped.get()); - if (ws != null) { - ws.close(1000, ""); - ws = null; - } - try { - while (!eventQueue.isEmpty()) { - Map event = eventQueue.poll(); - MyRunnable method = eventHandlers.get(event.get("method").toString()); - if (method != null) { - method.setMessage(event.get("params")); - method.run(); - } - } - } catch (Exception ignored) { - } - eventHandlers.clear(); - methodResults.clear(); - eventQueue.clear(); - if (occupant != null) occupant.onDisconnect(); - } - - public void setCallback(String event, MyRunnable callback) { - setCallback(event, callback, false); - } - - /** - * 绑定cdp event和回调方法 - * - * @param event 方法名称 - * @param callback 绑定到cdp event的回调方法 - * @param immediate 是否要立即处理的动作 - */ - public void setCallback(String event, MyRunnable callback, boolean immediate) { - Map handler = immediate ? immediateEventHandlers : eventHandlers; - if (callback != null) handler.put(event, callback); - else handler.remove(event); - } -} \ No newline at end of file diff --git a/java/src/main/java/com/ll/DrissonPage/base/Driver2.java b/java/src/main/java/com/ll/DrissonPage/base/Driver2.java deleted file mode 100644 index 8d2230e..0000000 --- a/java/src/main/java/com/ll/DrissonPage/base/Driver2.java +++ /dev/null @@ -1,423 +0,0 @@ -package com.ll.DrissonPage.base; - -/** - * 驱动 - * - * @author 陆 - * @address click - * @original DrissionPage - */ - -public class Driver2 { -// /** -// * 标签id -// */ -// @Getter -// private final String id; -// /** -// * 浏览器连接地址 -// */ -// @Getter -// private final String address; -// /** -// * 标签页类型 -// */ -// @Getter -// private final String type; -// private final boolean debug; -// private final String websocketUrl; -// private final AtomicInteger curId; -// /** -// * 会话返回值 -// */ -// private final AtomicReference webSocketMsg = new AtomicReference<>(null); -// private final Thread recvThread; -// private final Thread handleEventThread; -// @Getter -// -// private final AtomicBoolean stopped; -// private final BlockingQueue> eventQueue; -// private final BlockingQueue> immediateEventQueue; -// private final Map eventHandlers; -// private final Map immediateEventHandlers; -// private final Map>> methodResults; -// /** -// * 创建这个驱动的对象 -// */ -// @Getter -// @Setter -// private Occupant occupant; -// private boolean alertFlag; -// /** -// * 会话驱动 -// */ -// private WebSocket ws; -// private Thread handleImmediateEventThread; -// -// public Driver2(String tabId, String tabType, String address) { -// this(tabId, tabType, address, null); -// } -// -// /** -// * 驱动 -// * -// * @param tabId 标签id -// * @param tabType 标签页类型 -// * @param address 浏览器连接地址 -// */ -// public Driver2(String tabId, String tabType, String address, Occupant occupant) { -// this.id = tabId; -// this.address = address; -// this.type = tabType; -// this.occupant = occupant; -// this.debug = true; -// this.alertFlag = false; -// this.websocketUrl = "ws://" + address + "/devtools/" + tabType + "/" + tabId; -// this.curId = new AtomicInteger(0); -// this.ws = null; -// -// -// this.recvThread = new Thread(this::recvLoop); -// this.handleEventThread = new Thread(this::handleEventLoop); -// this.recvThread.setDaemon(true); -// this.handleEventThread.setDaemon(true); -// this.handleImmediateEventThread = null; -// -// this.stopped = new AtomicBoolean(); -// -// this.eventHandlers = new ConcurrentHashMap<>(); -// this.immediateEventHandlers = new ConcurrentHashMap<>(); -// this.methodResults = new ConcurrentHashMap<>(); -// this.eventQueue = new LinkedBlockingQueue<>(); -// this.immediateEventQueue = new LinkedBlockingQueue<>(); -// start(); -// } -// -// /** -// * 发送信息到浏览器,并返回浏览器返回的信息 -// * -// * @param message 发送给浏览器的数据 -// * @param timeout 超时时间,为null表示无时间 -// * @return 浏览器返回的数据 -// */ -// private JSONObject send(Map message, double timeout) { -// message = new HashMap<>(message); -// int wsId = curId.incrementAndGet(); -// message.put("id", wsId); -// String messageJson = JSON.toJSONString(message); -// -// if (this.debug) System.out.println("发->" + messageJson); -// //计算等待时间 -// long endTime = (long) (System.currentTimeMillis() + timeout * 1000L); -// LinkedBlockingQueue> value = new LinkedBlockingQueue<>(); -// methodResults.put(wsId, value); -// try { -// ws.send(messageJson); -// if (timeout == 0) { -// methodResults.remove(wsId); -// return new JSONObject(Map.of("id", wsId, "result", Map.of())); -// } -// } catch (WebsocketNotConnectedException e) { -// e.printStackTrace(); -// methodResults.remove(wsId); -// return new JSONObject(Map.of("error", Map.of("message", "connection disconnected"), "type", "connection_error")); -// } -// int i = 5; -// long endTimes = System.currentTimeMillis() + 1000L; -// while (!stopped.get()) { -// try { -// Map result = methodResults.get(wsId).poll(200, TimeUnit.MILLISECONDS); -// if (result == null && System.currentTimeMillis() < endTimes) continue; -// if (result == null && i > 0 && System.currentTimeMillis() > endTimes) { -// i--; -// endTimes = System.currentTimeMillis() + 1000L; -// ws.send(messageJson); -// continue; -// } -// methodResults.remove(wsId); -// if (result == null) throw new NullPointerException(); -// return new JSONObject(result); -// } catch (InterruptedException | NullPointerException | IllegalArgumentException e) { -//// e.printStackTrace(); -// String string = message.get("method").toString(); -// if (alertFlag && string.startsWith("Input.") || string.startsWith("Runtime.")) { -// return new JSONObject(Map.of("error", Map.of("message", "alert exists."), "type", "alert_exists")); -// } -// if (timeout > 0 && System.currentTimeMillis() > endTime) { -// methodResults.remove(wsId); -// return alertFlag ? new JSONObject(Map.of("error", Map.of("message", "alert exists."), "type", "alert_exists")) : new JSONObject(Map.of("error", Map.of("message", "timeout"), "type", "timeout")); -// } -// } -// } -// -// return new JSONObject(Map.of("error", Map.of("message", "connection disconnected"), "type", "connection_error")); -// } -// -// /** -// * 接收浏览器信息的守护线程方法 -// */ -// private void recvLoop() { -// while (!stopped.get()) { -// JSONObject msg; -// try { -// String andSet = webSocketMsg.getAndSet(null); -// if (andSet != null) { -// msg = JSONObject.parseObject(andSet); -// } else continue; -// } catch (Exception e) { -// if (stop()) return; -// return; -// -// } -// if (this.debug) System.out.println("<-收" + msg); -// -// if (msg.containsKey("method")) { -// if (msg.getString("method").startsWith("Page.javascriptDialog")) { -// alertFlag = msg.getString("method").endsWith("Opening"); -// } -// MyRunnable function = immediateEventHandlers.get(msg.getString("method")); -// if (function != null) { -// this.handleImmediateEvent(function, msg.getOrDefault("params", new HashMap<>())); -// } else { -// try { -// eventQueue.put(msg); -// } catch (InterruptedException e) { -// throw new RuntimeException(e); -// } -// } -// } else { -// int i = 1000; -// Integer integer = msg.getInteger("id"); -// while (i-- > 0 && integer != null && !methodResults.containsKey(integer)) { -// try { -// Thread.sleep(10); -// } catch (InterruptedException e) { -// throw new RuntimeException(e); -// } -// } -// if (methodResults.containsKey(integer)) { -// try { -// methodResults.get(integer).put(msg); -// } catch (InterruptedException e) { -// throw new RuntimeException(e); -// } -// } else if (this.debug) { -// System.out.println("未知错误->" + msg); -// -// } -// } -// -// } -// } -// -// /** -// * 当接收到浏览器信息,执行已绑定的方法 -// */ -// private void handleEventLoop() { -// while (!stopped.get()) { -// Map event; -// try { -// event = eventQueue.poll(1, TimeUnit.SECONDS); -// } catch (InterruptedException e) { -// continue; -// } -// -// if (event != null) { -// MyRunnable function = eventHandlers.get(event.get("method").toString()); -// if (function != null) { -// function.setMessage(event.get("params")); -// function.run(); -// } -// } -// this.eventQueue.poll(); -// -// } -// } -// -// private void handleImmediateEventLoop() { -// while (!stopped.get() && !immediateEventQueue.isEmpty()) { -// Map event; -// try { -// event = immediateEventQueue.poll(1, TimeUnit.SECONDS); -// } catch (InterruptedException e) { -// e.printStackTrace(); -// continue; -// } -// if (event != null) { -// MyRunnable function = immediateEventHandlers.get(event.get("method").toString()); -// if (function != null) { -// function.setMessage(event.get("params")); -// function.run(); -// } -// } -// -// } -// } -// -// /** -// * 处理立即执行的动作 -// * -// * @param function 要运行下方法 -// * @param params 方法参数 -// */ -// private void handleImmediateEvent(MyRunnable function, Object params) { -// Map func = new HashMap<>(); -// func.put("method", function); -// func.put("params", params); -// immediateEventQueue.add(func); -// -// if (handleImmediateEventThread == null || !handleImmediateEventThread.isAlive()) { -// handleImmediateEventThread = new Thread(this::handleImmediateEventLoop); -// handleImmediateEventThread.setDaemon(true); -// handleImmediateEventThread.start(); -// } -// } -// -// /** -// * 执行cdp方法 -// * -// * @param method 方法 -// * @return 执行结果 -// */ -// public Object run(String method) { -// return run(method, new HashMap<>()); -// } -// -// /** -// * 执行cdp方法 -// * -// * @param method 方法 -// * @param params 参数 -// * @return 执行结果 -// */ -// public Object run(String method, Map params) { -// if (stopped.get()) return Map.of("error", "connection disconnected", "type", "connection_error"); -// params = new HashMap<>(params); -// Object timeout1 = params.remove("_timeout"); -// double timeout = timeout1 != null ? Float.parseFloat(timeout1.toString()) : 30.0; -// -// JSONObject result = this.send(Map.of("method", method, "params", params), timeout); -// if (!result.containsKey("result") && result.containsKey("error")) { -// HashMap map = new HashMap<>(); -// map.put("error", result.getJSONObject("error").get("message")); -// map.put("type", result.getOrDefault("type", "call_method_error")); -// map.put("method", method); -// map.put("args", params); -// map.put("timeout", timeout); -// return JSON.toJSONString(map); -// } else { -// return JSON.toJSONString(result.get("result")); -// } -// } -// -// -// /** -// * 启动连接 -// */ -// private void start() { -// this.stopped.set(false); -// try { -// Request build = new Request(new HttpUrl("socket","","","",80,new ArrayList<>(),null,null,this.websocketUrl), "GET", Headers.of(), null, new HashMap<>()).newBuilder().url(this.websocketUrl).build(); -// OkHttpClient okHttpClient = new OkHttpClient(); -// ws = okHttpClient.newWebSocket(build, new WebSocketListener() { -// @Override -// public void onClosed(@NotNull WebSocket webSocket, int code, @NotNull String reason) { -// // 关闭事件处理 -// stop(); -// super.onClosed(webSocket, code, reason); -// } -// -// @Override -// public void onClosing(@NotNull WebSocket webSocket, int code, @NotNull String reason) { -// super.onClosing(webSocket, code, reason); -// } -// -// @Override -// public void onFailure(@NotNull WebSocket webSocket, @NotNull Throwable t, @Nullable Response response) { -// super.onFailure(webSocket, t, response); -// } -// -// @Override -// public void onMessage(@NotNull WebSocket webSocket, @NotNull String text) { -// webSocketMsg.set(text); -// super.onMessage(webSocket, text); -// } -// -// @Override -// public void onMessage(@NotNull WebSocket webSocket, @NotNull ByteString bytes) { -// super.onMessage(webSocket, bytes); -// } -// -// @Override -// public void onOpen(@NotNull WebSocket webSocket, @NotNull Response response) { -// super.onOpen(webSocket, response); -// } -// -// }); -// recvThread.start(); -// handleEventThread.start(); -// } catch (Exception e) { -// e.printStackTrace(); -// stop(); -// } -// } -// -// /** -// * 中断连接 -// */ -// public boolean stop() { -// stop1(); -// while (this.recvThread.isAlive() || this.handleEventThread.isAlive()) { -// try { -// Thread.sleep(100); -// } catch (InterruptedException e) { -// throw new RuntimeException(e); -// } -// } -// return true; -// } -// -// /** -// * 中断连接 -// */ -// private void stop1() { -// if (stopped.get()) return; -// stopped.set(true); -// if (ws != null) { -// ws.close(1000, ""); -// ws = null; -// } -// try { -// while (!eventQueue.isEmpty()) { -// Map event = eventQueue.poll(); -// MyRunnable method = eventHandlers.get(event.get("method").toString()); -// if (method != null) { -// method.setMessage(event.get("params")); -// method.run(); -// } -// } -// } catch (Exception ignored) { -// } -// eventHandlers.clear(); -// methodResults.clear(); -// eventQueue.clear(); -// if (occupant != null) occupant.onDisconnect(); -// } -// -// public void setCallback(String event, MyRunnable callback) { -// setCallback(event, callback, false); -// } -// -// /** -// * 绑定cdp event和回调方法 -// * -// * @param event 方法名称 -// * @param callback 绑定到cdp event的回调方法 -// * @param immediate 是否要立即处理的动作 -// */ -// public void setCallback(String event, MyRunnable callback, boolean immediate) { -// Map handler = immediate ? immediateEventHandlers : eventHandlers; -// if (callback != null) handler.put(event, callback); -// else handler.remove(event); -// } -} \ No newline at end of file diff --git a/java/src/main/java/com/ll/DrissonPage/base/Driver3.java b/java/src/main/java/com/ll/DrissonPage/base/Driver3.java deleted file mode 100644 index e7b385c..0000000 --- a/java/src/main/java/com/ll/DrissonPage/base/Driver3.java +++ /dev/null @@ -1,426 +0,0 @@ -package com.ll.DrissonPage.base; - -/** - * 驱动 - * - * @author 陆 - * @address click - * @original DrissionPage - */ - -public class Driver3 { -// /** -// * 标签id -// */ -// @Getter -// private final String id; -// /** -// * 浏览器连接地址 -// */ -// @Getter -// private final String address; -// /** -// * 标签页类型 -// */ -// @Getter -// private final String type; -// private final boolean debug; -// private final String websocketUrl; -// private final AtomicInteger curId; -// /** -// * 会话返回值 -// */ -// private final AtomicReference webSocketMsg = new AtomicReference<>(null); -// private final Thread recvThread; -// private final Thread handleEventThread; -// @Getter -// -// private final AtomicBoolean stopped; -// private final BlockingQueue> eventQueue; -// private final BlockingQueue> immediateEventQueue; -// private final Map eventHandlers; -// private final Map immediateEventHandlers; -// private final Map>> methodResults; -// /** -// * 创建这个驱动的对象 -// */ -// @Getter -// @Setter -// private Occupant occupant; -// private boolean alertFlag; -// /** -// * 会话驱动 -// */ -// private WebSocketClient ws; -// private Future webSocketSession; -// private Thread handleImmediateEventThread; -// -// public Driver3(String tabId, String tabType, String address) { -// this(tabId, tabType, address, null); -// } -// -// /** -// * 驱动 -// * -// * @param tabId 标签id -// * @param tabType 标签页类型 -// * @param address 浏览器连接地址 -// */ -// public Driver3(String tabId, String tabType, String address, Occupant occupant) { -// this.id = tabId; -// this.address = address; -// this.type = tabType; -// this.occupant = occupant; -// this.debug = true; -// this.alertFlag = false; -// this.websocketUrl = "ws://" + address + "/devtools/" + tabType + "/" + tabId; -// this.curId = new AtomicInteger(0); -// this.ws = null; -// this.recvThread = new Thread(this::recvLoop); -// this.handleEventThread = new Thread(this::handleEventLoop); -// this.recvThread.setDaemon(true); -// this.handleEventThread.setDaemon(true); -// this.handleImmediateEventThread = null; -// -// this.stopped = new AtomicBoolean(); -// -// this.eventHandlers = new ConcurrentHashMap<>(); -// this.immediateEventHandlers = new ConcurrentHashMap<>(); -// this.methodResults = new ConcurrentHashMap<>(); -// this.eventQueue = new LinkedBlockingQueue<>(); -// this.immediateEventQueue = new LinkedBlockingQueue<>(); -// start(); -// } -// -// /** -// * 发送信息到浏览器,并返回浏览器返回的信息 -// * -// * @param message 发送给浏览器的数据 -// * @param timeout 超时时间,为null表示无时间 -// * @return 浏览器返回的数据 -// */ -// private JSONObject send(Map message, double timeout) { -// message = new HashMap<>(message); -// int wsId = curId.incrementAndGet(); -// message.put("id", wsId); -// String messageJson = JSON.toJSONString(message); -// -// if (this.debug) System.out.println("发->" + messageJson); -// //计算等待时间 -// long endTime = (long) (System.currentTimeMillis() + timeout * 1000L); -// LinkedBlockingQueue> value = new LinkedBlockingQueue<>(); -// methodResults.put(wsId, value); -// try { -// if (ws.isStopping()) return new JSONObject(); -// webSocketSession.get().getRemote().sendString(messageJson); -// if (timeout == 0) { -// methodResults.remove(wsId); -// return new JSONObject(Map.of("id", wsId, "result", Map.of())); -// } -// } catch (WebsocketNotConnectedException | IOException | InterruptedException | ExecutionException e) { -// e.printStackTrace(); -// methodResults.remove(wsId); -// return new JSONObject(Map.of("error", Map.of("message", "connection disconnected"), "type", "connection_error")); -// } -// int i = 5; -// long endTimes = System.currentTimeMillis() + 1000L; -// while (!stopped.get()) { -// try { -// Map result = methodResults.get(wsId).poll(2000, TimeUnit.MILLISECONDS); -// if (result == null && System.currentTimeMillis() < endTimes) continue; -// if (result == null && i > 0 && System.currentTimeMillis() > endTimes) { -// System.out.println("丢包:->" + messageJson); -// i--; -// endTimes = System.currentTimeMillis() + 1000L; -// if (ws.isStopping()) return new JSONObject(); -// webSocketSession.get().getRemote().sendString(messageJson); -// continue; -// } -// methodResults.remove(wsId); -// if (result == null) throw new NullPointerException(); -// return new JSONObject(result); -// } catch (InterruptedException | NullPointerException | IllegalArgumentException | IOException | -// ExecutionException e) { -//// e.printStackTrace(); -// String string = message.get("method").toString(); -// if (alertFlag && string.startsWith("Input.") || string.startsWith("Runtime.")) { -// return new JSONObject(Map.of("error", Map.of("message", "alert exists."), "type", "alert_exists")); -// } -// if (timeout > 0 && System.currentTimeMillis() > endTime) { -// methodResults.remove(wsId); -// return alertFlag ? new JSONObject(Map.of("error", Map.of("message", "alert exists."), "type", "alert_exists")) : new JSONObject(Map.of("error", Map.of("message", "timeout"), "type", "timeout")); -// } -// } -// } -// -// return new JSONObject(Map.of("error", Map.of("message", "connection disconnected"), "type", "connection_error")); -// } -// -// /** -// * 接收浏览器信息的守护线程方法 -// */ -// private void recvLoop() { -// while (!stopped.get()) { -// JSONObject msg; -// try { -// String andSet = webSocketMsg.getAndSet(null); -// if (andSet != null) { -// msg = JSONObject.parseObject(andSet); -// } else continue; -// } catch (Exception e) { -// if (stop()) return; -// return; -// -// } -// if (this.debug) System.out.println("<-收" + msg); -// -// if (msg.containsKey("method")) { -// if (msg.getString("method").startsWith("Page.javascriptDialog")) { -// alertFlag = msg.getString("method").endsWith("Opening"); -// } -// MyRunnable function = immediateEventHandlers.get(msg.getString("method")); -// if (function != null) { -// this.handleImmediateEvent(function, msg.getOrDefault("params", new HashMap<>())); -// } else { -// try { -// eventQueue.put(msg); -// } catch (InterruptedException e) { -// throw new RuntimeException(e); -// } -// } -// } else { -// int i = 1000; -// Integer integer = msg.getInteger("id"); -// while (i-- > 0 && integer != null && !methodResults.containsKey(integer)) { -// try { -// Thread.sleep(10); -// } catch (InterruptedException e) { -// throw new RuntimeException(e); -// } -// } -// if (methodResults.containsKey(integer)) { -// try { -// methodResults.get(integer).put(msg); -// } catch (InterruptedException e) { -// throw new RuntimeException(e); -// } -// } else if (this.debug) { -// System.out.println("未知错误->" + msg); -// -// } -// } -// -// } -// } -// -// /** -// * 当接收到浏览器信息,执行已绑定的方法 -// */ -// private void handleEventLoop() { -// while (!stopped.get()) { -// Map event; -// try { -// event = eventQueue.poll(1, TimeUnit.SECONDS); -// } catch (InterruptedException e) { -// continue; -// } -// -// if (event != null) { -// MyRunnable function = eventHandlers.get(event.get("method").toString()); -// if (function != null) { -// function.setMessage(event.get("params")); -// function.run(); -// } -// } -// this.eventQueue.poll(); -// -// } -// } -// -// private void handleImmediateEventLoop() { -// while (!stopped.get() && !immediateEventQueue.isEmpty()) { -// Map event; -// try { -// event = immediateEventQueue.poll(1, TimeUnit.SECONDS); -// } catch (InterruptedException e) { -// e.printStackTrace(); -// continue; -// } -// if (event != null) { -// MyRunnable function = immediateEventHandlers.get(event.get("method").toString()); -// if (function != null) { -// function.setMessage(event.get("params")); -// function.run(); -// } -// } -// -// } -// } -// -// /** -// * 处理立即执行的动作 -// * -// * @param function 要运行下方法 -// * @param params 方法参数 -// */ -// private void handleImmediateEvent(MyRunnable function, Object params) { -// Map func = new HashMap<>(); -// func.put("method", function); -// func.put("params", params); -// immediateEventQueue.add(func); -// -// if (handleImmediateEventThread == null || !handleImmediateEventThread.isAlive()) { -// handleImmediateEventThread = new Thread(this::handleImmediateEventLoop); -// handleImmediateEventThread.setDaemon(true); -// handleImmediateEventThread.start(); -// } -// } -// -// /** -// * 执行cdp方法 -// * -// * @param method 方法 -// * @return 执行结果 -// */ -// public Object run(String method) { -// return run(method, new HashMap<>()); -// } -// -// /** -// * 执行cdp方法 -// * -// * @param method 方法 -// * @param params 参数 -// * @return 执行结果 -// */ -// public Object run(String method, Map params) { -// if (stopped.get()) return Map.of("error", "connection disconnected", "type", "connection_error"); -// params = new HashMap<>(params); -// Object timeout1 = params.remove("_timeout"); -// double timeout = timeout1 != null ? Float.parseFloat(timeout1.toString()) : 30.0; -// -// JSONObject result = this.send(Map.of("method", method, "params", params), timeout); -// if (!result.containsKey("result") && result.containsKey("error")) { -// HashMap map = new HashMap<>(); -// map.put("error", result.getJSONObject("error").get("message")); -// map.put("type", result.getOrDefault("type", "call_method_error")); -// map.put("method", method); -// map.put("args", params); -// map.put("timeout", timeout); -// return JSON.toJSONString(map); -// } else { -// return JSON.toJSONString(result.get("result")); -// } -// } -// -// -// /** -// * 启动连接 -// */ -// private void start() { -// this.stopped.set(false); -// try { -//// Request build = new Request(new HttpUrl("socket","","","",80,new ArrayList<>(),null,null,this.websocketUrl), "GET", Headers.of(), null, new HashMap<>()).newBuilder().url(this.websocketUrl).build(); -//// OkHttpClient okHttpClient = new OkHttpClient(); -// ws = new WebSocketClient(); -// ws.setConnectTimeout(60_000); -// ws.start(); -// webSocketSession = ws.connect(new WebSocketAdapter() { -// @Override -// public void onWebSocketClose(int statusCode, String reason) { -// stop(); -// super.onWebSocketClose(statusCode, reason); -// } -// -// @Override -// public void onWebSocketConnect(Session sess) { -// super.onWebSocketConnect(sess); -// } -// -// @Override -// public void onWebSocketError(Throwable cause) { -// super.onWebSocketError(cause); -// } -// -// @Override -// public boolean isConnected() { -// return super.isConnected(); -// } -// -// @Override -// public void onWebSocketText(String message) { -// webSocketMsg.set(message); -// } -// -// }, URI.create(this.websocketUrl)); -// recvThread.start(); -// handleEventThread.start(); -// } catch (Exception e) { -// e.printStackTrace(); -// stop(); -// } -// } -// -// /** -// * 中断连接 -// */ -// public boolean stop() { -// stop1(); -// while (this.recvThread.isAlive() || this.handleEventThread.isAlive()) { -// try { -// Thread.sleep(100); -// } catch (InterruptedException e) { -// throw new RuntimeException(e); -// } -// } -// return true; -// } -// -// /** -// * 中断连接 -// */ -// private void stop1() { -// if (stopped.get()) return; -// stopped.set(true); -// if (ws != null) { -// try { -// ws.stop(); -// } catch (Exception e) { -// throw new RuntimeException(e); -// } -// ws = null; -// } -// try { -// while (!eventQueue.isEmpty()) { -// Map event = eventQueue.poll(); -// MyRunnable method = eventHandlers.get(event.get("method").toString()); -// if (method != null) { -// method.setMessage(event.get("params")); -// method.run(); -// } -// } -// } catch (Exception ignored) { -// } -// eventHandlers.clear(); -// methodResults.clear(); -// eventQueue.clear(); -// if (occupant != null) occupant.onDisconnect(); -// } -// -// public void setCallback(String event, MyRunnable callback) { -// setCallback(event, callback, false); -// } -// -// /** -// * 绑定cdp event和回调方法 -// * -// * @param event 方法名称 -// * @param callback 绑定到cdp event的回调方法 -// * @param immediate 是否要立即处理的动作 -// */ -// public void setCallback(String event, MyRunnable callback, boolean immediate) { -// Map handler = immediate ? immediateEventHandlers : eventHandlers; -// if (callback != null) handler.put(event, callback); -// else handler.remove(event); -// } -} \ No newline at end of file diff --git a/java/src/main/java/com/ll/DrissonPage/base/Driver_org_webSocket.java b/java/src/main/java/com/ll/DrissonPage/base/Driver_org_webSocket.java deleted file mode 100644 index 68780b2..0000000 --- a/java/src/main/java/com/ll/DrissonPage/base/Driver_org_webSocket.java +++ /dev/null @@ -1,424 +0,0 @@ -package com.ll.DrissonPage.base; - -/** - * 驱动 org.java-websocket - * - * @author 陆 - * @address click - * @original DrissionPage - */ - -public class Driver_org_webSocket { -// /** -// * 标签id -// */ -// @Getter -// private final String id; -// /** -// * 浏览器连接地址 -// */ -// @Getter -// private final String address; -// /** -// * 标签页类型 -// */ -// @Getter -// private final String type; -// private final boolean debug; -// private final String websocketUrl; -// private final AtomicInteger curId; -// /** -// * 会话返回值 -// */ -// private final AtomicReference webSocketMsg = new AtomicReference<>(null); -// private final Thread recvThread; -// private final Thread handleEventThread; -// @Getter -// -// private final AtomicBoolean stopped; -// private final BlockingQueue> eventQueue; -// private final BlockingQueue> immediateEventQueue; -// private final Map eventHandlers; -// private final Map immediateEventHandlers; -// private final Map>> methodResults; -// /** -// * 创建这个驱动的对象 -// */ -// @Getter -// @Setter -// private Occupant occupant; -// private boolean alertFlag; -// /** -// * 会话驱动 -// */ -// private WebSocketClient ws; -// private Thread handleImmediateEventThread; -// -// public Driver(String tabId, String tabType, String address) { -// this(tabId, tabType, address, null); -// } -// -// /** -// * 驱动 -// * -// * @param tabId 标签id -// * @param tabType 标签页类型 -// * @param address 浏览器连接地址 -// */ -// public Driver(String tabId, String tabType, String address, Occupant occupant) { -// this.id = tabId; -// this.address = address; -// this.type = tabType; -// this.occupant = occupant; -// this.debug = true; -// this.alertFlag = false; -// this.websocketUrl = "ws://" + address + "/devtools/" + tabType + "/" + tabId; -// this.curId = new AtomicInteger(0); -// this.ws = null; -// -// -// this.recvThread = new Thread(this::recvLoop); -// this.handleEventThread = new Thread(this::handleEventLoop); -// this.recvThread.setDaemon(true); -// this.handleEventThread.setDaemon(true); -// this.handleImmediateEventThread = null; -// -// this.stopped = new AtomicBoolean(); -// -// this.eventHandlers = new ConcurrentHashMap<>(); -// this.immediateEventHandlers = new ConcurrentHashMap<>(); -// this.methodResults = new ConcurrentHashMap<>(); -// this.eventQueue = new LinkedBlockingQueue<>(); -// this.immediateEventQueue = new LinkedBlockingQueue<>(); -// start(); -// } -// -// /** -// * 发送信息到浏览器,并返回浏览器返回的信息 -// * -// * @param message 发送给浏览器的数据 -// * @param timeout 超时时间,为null表示无时间 -// * @return 浏览器返回的数据 -// */ -// private JSONObject send(Map message, double timeout) { -// message = new HashMap<>(message); -// int wsId = curId.incrementAndGet(); -// message.put("id", wsId); -// String messageJson = JSON.toJSONString(message); -// -// if (this.debug) System.out.println("发->" + messageJson); -// //计算等待时间 -// long endTime = (long) (System.currentTimeMillis() + timeout * 1000L); -// LinkedBlockingQueue> value = new LinkedBlockingQueue<>(); -// methodResults.put(wsId, value); -// try { -// ws.send(messageJson); -// if (timeout == 0) { -// methodResults.remove(wsId); -// return new JSONObject(Map.of("id", wsId, "result", Map.of())); -// } -// } catch (WebsocketNotConnectedException e) { -// e.printStackTrace(); -// methodResults.remove(wsId); -// return new JSONObject(Map.of("error", Map.of("message", "connection disconnected"), "type", "connection_error")); -// } -// int i = 5; -// long endTimes = System.currentTimeMillis() + 1000L; -// while (!stopped.get()) { -// try { -// Map result = methodResults.get(wsId).poll(10_000, TimeUnit.MILLISECONDS); -// if (result == null && System.currentTimeMillis() < endTimes) continue; -// if (result == null && i > 0 && System.currentTimeMillis() > endTimes) { -// i--; -// endTimes = System.currentTimeMillis() + 1000L; -// System.out.println("超时丢包:->" + messageJson); -// ws.send(messageJson); -// continue; -// } -// methodResults.remove(wsId); -// if (result == null) throw new NullPointerException(); -// return new JSONObject(result); -// } catch (InterruptedException | NullPointerException | IllegalArgumentException e) { -//// e.printStackTrace(); -// String string = message.get("method").toString(); -// if (alertFlag && string.startsWith("Input.") || string.startsWith("Runtime.")) { -// return new JSONObject(Map.of("error", Map.of("message", "alert exists."), "type", "alert_exists")); -// } -// if (timeout > 0 && System.currentTimeMillis() > endTime) { -// methodResults.remove(wsId); -// return alertFlag ? new JSONObject(Map.of("error", Map.of("message", "alert exists."), "type", "alert_exists")) : new JSONObject(Map.of("error", Map.of("message", "timeout"), "type", "timeout")); -// } -// } -// } -// -// return new JSONObject(Map.of("error", Map.of("message", "connection disconnected"), "type", "connection_error")); -// } -// -// /** -// * 接收浏览器信息的守护线程方法 -// */ -// private void recvLoop() { -// while (!stopped.get()) { -// JSONObject msg; -// try { -// String andSet = webSocketMsg.getAndSet(null); -// if (andSet != null) { -// msg = JSONObject.parseObject(andSet); -// } else continue; -// } catch (Exception e) { -// if (stop()) return; -// return; -// -// } -// if (this.debug) System.out.println("<-收" + msg); -// -// if (msg.containsKey("method")) { -// if (msg.getString("method").startsWith("Page.javascriptDialog")) { -// alertFlag = msg.getString("method").endsWith("Opening"); -// } -// MyRunnable function = immediateEventHandlers.get(msg.getString("method")); -// if (function != null) { -// this.handleImmediateEvent(function, msg.getOrDefault("params", new HashMap<>())); -// } else { -// try { -// eventQueue.put(msg); -// } catch (InterruptedException e) { -// throw new RuntimeException(e); -// } -// } -// } else { -// int i = 1000; -// Integer integer = msg.getInteger("id"); -// while (i-- > 0 && integer != null && !methodResults.containsKey(integer)) { -// try { -// Thread.sleep(10); -// } catch (InterruptedException e) { -// throw new RuntimeException(e); -// } -// } -// if (methodResults.containsKey(integer)) { -// try { -// methodResults.get(integer).put(msg); -// } catch (InterruptedException e) { -// throw new RuntimeException(e); -// } -// } else if (this.debug) { -// System.out.println("未知错误->" + msg); -// -// } -// } -// -// } -// } -// -// /** -// * 当接收到浏览器信息,执行已绑定的方法 -// */ -// private void handleEventLoop() { -// while (!stopped.get()) { -// Map event; -// try { -// event = eventQueue.poll(1, TimeUnit.SECONDS); -// } catch (InterruptedException e) { -// continue; -// } -// -// if (event != null) { -// MyRunnable function = eventHandlers.get(event.get("method").toString()); -// if (function != null) { -// function.setMessage(event.get("params")); -// function.run(); -// } -// } -// this.eventQueue.poll(); -// -// } -// } -// -// private void handleImmediateEventLoop() { -// while (!stopped.get() && !immediateEventQueue.isEmpty()) { -// Map event; -// try { -// event = immediateEventQueue.poll(1, TimeUnit.SECONDS); -// } catch (InterruptedException e) { -// e.printStackTrace(); -// continue; -// } -// if (event != null) { -// MyRunnable function = immediateEventHandlers.get(event.get("method").toString()); -// if (function != null) { -// function.setMessage(event.get("params")); -// function.run(); -// } -// } -// -// } -// } -// -// /** -// * 处理立即执行的动作 -// * -// * @param function 要运行下方法 -// * @param params 方法参数 -// */ -// private void handleImmediateEvent(MyRunnable function, Object params) { -// Map func = new HashMap<>(); -// func.put("method", function); -// func.put("params", params); -// immediateEventQueue.add(func); -// -// if (handleImmediateEventThread == null || !handleImmediateEventThread.isAlive()) { -// handleImmediateEventThread = new Thread(this::handleImmediateEventLoop); -// handleImmediateEventThread.setDaemon(true); -// handleImmediateEventThread.start(); -// } -// } -// -// /** -// * 执行cdp方法 -// * -// * @param method 方法 -// * @return 执行结果 -// */ -// public Object run(String method) { -// return run(method, new HashMap<>()); -// } -// -// /** -// * 执行cdp方法 -// * -// * @param method 方法 -// * @param params 参数 -// * @return 执行结果 -// */ -// public Object run(String method, Map params) { -// if (stopped.get()) return Map.of("error", "connection disconnected", "type", "connection_error"); -// params = new HashMap<>(params); -// Object timeout1 = params.remove("_timeout"); -// double timeout = timeout1 != null ? Float.parseFloat(timeout1.toString()) : 30.0; -// -// JSONObject result = this.send(Map.of("method", method, "params", params), timeout); -// if (!result.containsKey("result") && result.containsKey("error")) { -// HashMap map = new HashMap<>(); -// map.put("error", result.getJSONObject("error").get("message")); -// map.put("type", result.getOrDefault("type", "call_method_error")); -// map.put("method", method); -// map.put("args", params); -// map.put("timeout", timeout); -// return JSON.toJSONString(map); -// } else { -// return JSON.toJSONString(result.get("result")); -// } -// } -// -// -// /** -// * 启动连接 -// */ -// private void start() { -// this.stopped.set(false); -// try { -// ws = new WebSocketClient(new URI(websocketUrl)) { -// @Override -// public void onOpen(ServerHandshake handshakeData) { -// // 处理 WebSocket 打开事件 -// } -// -// @Override -// public void onMessage(String message) { -// //处理返回数据 -// webSocketMsg.set(message); -// } -// -// -// @Override -// public void onClose(int code, String reason, boolean remote) { -// -// System.out.println("关闭" + reason); -// // 关闭事件处理 -// stop(); -// } -// -// @Override -// public void onError(Exception ex) { -// System.out.println("错误" + ex.getMessage()); -// // 错误事件处理 -//// stop(); -// } -// }; -// ws.setConnectionLostTimeout(60); -// ws.connect(); -// //需要睡0.1秒让其等待 -// while (ws != null && !ws.getReadyState().equals(ReadyState.OPEN)) { -// Thread.sleep(10); -// } -// recvThread.start(); -// handleEventThread.start(); -// } catch (Exception e) { -// e.printStackTrace(); -// stop(); -// } -// } -// -// /** -// * 中断连接 -// */ -// public boolean stop() { -// stop1(); -// while (this.recvThread.isAlive() || this.handleEventThread.isAlive()) { -// try { -// Thread.sleep(100); -// } catch (InterruptedException e) { -// throw new RuntimeException(e); -// } -// } -// return true; -// } -// -// /** -// * 中断连接 -// */ -// private void stop1() { -// if (stopped.get()) { -// return; -// } -// stopped.set(true); -// if (ws != null) { -// ws.close(); -// ws = null; -// } -// -// try { -// while (!eventQueue.isEmpty()) { -// Map
, T extends DrissionElement, ?>> extends BaseElement
{ - - public DrissionElement(P page) { - super(page); - this.setType("DrissionElement"); - } - - /** - * 返回href或者src绝对的url - */ - public String link() { - String href = this.attr("href"); - return href == null ? this.attr("src") : href; - } - - /** - * 返回css path路径 - */ - public String cssPath() { - return this.getElePath(ElePathMode.CSS); - } - - /** - * 返回xpath路径 - */ - public String xpath() { - return this.getElePath(ElePathMode.XPATH); - } - - /** - * 返回元素注释文本组成的列表 - */ - public List comments() { - return this.eles("xpath:.//comment()"); - } - - /** - * 返回元素内所有直接子节点的文本,包括元素和文本节点 - * - * @return 文本列表 - */ - public List texts() { - return this.texts(false); - } - - /** - * 返回元素内所有直接子节点的文本,包括元素和文本节点 - * - * @param textNodeOnly 是否只返回文本节点 - * @return 文本列表 - */ - public List texts(boolean textNodeOnly) { - List texts = new ArrayList<>(); - if (textNodeOnly) { - this.eles("xpath:/text()").forEach(a -> texts.add(a.text())); - } else { - this.eles("xpath:/text() | *").forEach(a -> texts.add(a.text())); - } - return texts.stream().filter(text -> text != null && !Pattern.compile("[\r\n\t ]").matcher(text).replaceAll("").isEmpty()).map(text -> Web.formatHtml(text.trim().replaceAll("[\r\n]", ""))).collect(Collectors.toList()); - } - - /** - * 返回上面某一级父元素 用查询语法定位 - * - * @param by 查询选择器 - * @param index 选择第几个结果 - * @return 上级元素对象 - */ - @Override - public T parent(By by, Integer index) { - by = Locator.getLoc(by, true, false); - if (!by.getName().equals(BySelect.CSS_SELECTOR)) - throw new IllegalArgumentException("此css selector语法不受支持,请换成xpath"); - String loc = "xpath:./ancestor::" + by.getValue().replaceAll("^[.\\s/]+", "") + "[" + index + "]"; - return _ele(loc, null, 1, true, false, "parent()").get(0); - } - - /** - * 返回上面某一级父元素,指定层数 - * - * @param level 第几级父元素 - * @return 上级元素对象 - */ - @Override - public T parent(Integer level) { - String loc = "xpath:./ancestor::*[" + level + "]"; - return _ele(loc, null, 1, true, false, "parent()").get(0); - } - - /** - * 返回上面某一级父元素 用查询语法定位 - * - * @param loc 定位符 - * @param index 选择第几个结果 - * @return 上级元素对象 - */ - @Override - public T parent(String loc, Integer index) { - By loc1 = Locator.getLoc(loc, true, false); - if (!loc1.getName().equals(BySelect.CSS_SELECTOR)) - throw new IllegalArgumentException("此css selector语法不受支持,请换成xpath"); - loc = "xpath:./ancestor::" + loc1.getValue().replaceAll("^[.\\s/]+", "") + "[" + index + "]"; - return _ele(loc, null, 1, true, false, "parent()").get(0); - } - - /** - * 返回直接子元素元素或节点组成的列表,可用查询语法筛选 - * - * @param index 第几个查询结果,1开始 - * @return 直接子元素或节点文本组成的列表 - */ - public T child(Integer index) { - return child(index, null); - } - - /** - * 返回直接子元素元素或节点组成的列表,可用查询语法筛选 - * - * @param index 第几个查询结果,1开始 - * @param timeout 查找节点的超时时间 - * @return 直接子元素或节点文本组成的列表 - */ - public T child(Integer index, Double timeout) { - return child(index, timeout, true); - } - - /** - * 返回直接子元素元素或节点组成的列表,可用查询语法筛选 - * - * @param index 第几个查询结果,1开始 - * @param timeout 查找节点的超时时间 - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 直接子元素或节点文本组成的列表 - */ - public T child(Integer index, Double timeout, Boolean eleOnly) { - String loc = eleOnly ? "*" : "node()"; - List ts = this._ele("xpath./" + loc, timeout, index, true, false, null); - if (!ts.isEmpty()) { - return ts.get(0); - } else if (Settings.raiseWhenEleNotFound) { - throw new ElementNotFoundError("child()", Map.of("loc", "", "index", index, "eleOnly", eleOnly)); - } - return null; - } - - /** - * 返回直接子元素元素或节点组成的列表,可用查询语法筛选 - * - * @return 直接子元素或节点文本组成的列表 - */ - public T child() { - return child(""); - } - - /** - * 返回直接子元素元素或节点组成的列表,可用查询语法筛选 - * - * @param loc 查询语句 - * @return 直接子元素或节点文本组成的列表 - */ - public T child(String loc) { - return child(loc.isEmpty() ? null : loc, 1); - } - - /** - * 返回直接子元素元素或节点组成的列表,可用查询语法筛选 - * - * @param loc 查询语句 - * @param index 第几个查询结果,1开始 - * @return 直接子元素或节点文本组成的列表 - */ - public T child(String loc, Integer index) { - return child(loc, index, null); - } - - /** - * 返回直接子元素元素或节点组成的列表,可用查询语法筛选 - * - * @param loc 查询语句 - * @param index 第几个查询结果,1开始 - * @param timeout 查找节点的超时时间 - * @return 直接子元素或节点文本组成的列表 - */ - public T child(String loc, Integer index, Double timeout) { - return child(loc, index, timeout, true); - } - - /** - * 返回直接子元素元素或节点组成的列表,可用查询语法筛选 - * - * @param loc 查询语句 - * @param index 第几个查询结果,1开始 - * @param timeout 查找节点的超时时间 - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 直接子元素或节点文本组成的列表 - */ - public T child(String loc, Integer index, Double timeout, Boolean eleOnly) { - if (loc == null || loc.isEmpty()) { - loc = eleOnly ? "*" : "node()"; - } else { - By by = Locator.getLoc(loc, true, false); - if (by.getName().equals(BySelect.CSS_SELECTOR)) - throw new IllegalArgumentException("此css selector语法不受支持,请换成xpath。"); - else loc = by.getValue().replaceAll("^[.\\s/]+", ""); - } - - List ts = this._ele("xpath./" + loc, timeout, index, true, false, null); - if (!ts.isEmpty()) { - return ts.get(0); - } else if (Settings.raiseWhenEleNotFound) { - throw new ElementNotFoundError("child()", Map.of("loc", loc, "index", index, "eleOnly", eleOnly)); - } - return null; - } - - /** - * 返回直接子元素元素或节点组成的列表,可用查询语法筛选 - * - * @param by 查询语句 - * @return 直接子元素或节点文本组成的列表 - */ - public T child(By by) { - return child(by, 1); - } - - /** - * 返回直接子元素元素或节点组成的列表,可用查询语法筛选 - * - * @param by 查询语句 - * @param index 第几个查询结果,1开始 - * @return 直接子元素或节点文本组成的列表 - */ - public T child(By by, Integer index) { - return child(by, index, null); - } - - /** - * 返回直接子元素元素或节点组成的列表,可用查询语法筛选 - * - * @param by 查询语句 - * @param index 第几个查询结果,1开始 - * @param timeout 查找节点的超时时间 - * @return 直接子元素或节点文本组成的列表 - */ - public T child(By by, Integer index, Double timeout) { - return child(by, index, timeout, true); - } - - /** - * 返回直接子元素元素或节点组成的列表,可用查询语法筛选 - * - * @param by 查询语句 - * @param index 第几个查询结果,1开始 - * @param timeout 查找节点的超时时间 - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 直接子元素或节点文本组成的列表 - */ - public T child(By by, Integer index, Double timeout, Boolean eleOnly) { - String loc; - if (by == null) { - loc = eleOnly ? "*" : "node()"; - } else { - by = Locator.getLoc(by, true, false); - if (by.getName().equals(BySelect.CSS_SELECTOR)) - throw new IllegalArgumentException("此css selector语法不受支持,请换成xpath。"); - else loc = by.getValue().replaceAll("^[.\\s/]+", ""); - } - - List ts = this._ele("xpath./" + loc, timeout, index, true, false, null); - if (!ts.isEmpty()) { - return ts.get(0); - } else if (Settings.raiseWhenEleNotFound) { - throw new ElementNotFoundError("child()", Map.of("loc", loc, "index", index, "eleOnly", eleOnly)); - } - return null; - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @return 兄弟元素 - */ - public T prev() { - return prev(""); - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param loc 用于筛选的查询语法 - * @return 兄弟元素 - */ - public T prev(String loc) { - return prev(loc.isEmpty() ? null : loc, null); - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param loc 用于筛选的查询语法 - * @param index 前面第几个查询结果,0开始 - * @return 兄弟元素 - */ - - public T prev(String loc, Integer index) { - return prev(loc, index, null); - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param loc 用于筛选的查询语法 - * @param index 前面第几个查询结果,0开始 - * @param timeout 查找节点的超时时间(秒) - * @return 兄弟元素 - */ - public T prev(String loc, Integer index, Double timeout) { - return prev(loc, index, timeout, true); - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param loc 用于筛选的查询语法 - * @param index 前面第几个查询结果,1开始 - * @param timeout 查找节点的超时时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 兄弟元素 - */ - public T prev(String loc, Integer index, Double timeout, Boolean eleOnly) { - return this.getRelative("prev()", "preceding", true, loc, index, timeout, eleOnly); - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param by 用于筛选的元素 - * @return 兄弟元素 - */ - public T prev(By by) { - return prev(by, 1); - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param by 用于筛选的元素 - * @param index 前面第几个查询结果,0开始 - * @return 兄弟元素 - */ - public T prev(By by, Integer index) { - return prev(by, index, null); - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param by 用于筛选的元素 - * @param index 前面第几个查询结果,0开始 - * @param timeout 查找节点的超时时间(秒) - * @return 兄弟元素 - */ - public T prev(By by, Integer index, Double timeout) { - return prev(by, index, timeout, true); - } - - public T prev(By by, Integer index, Double timeout, Boolean eleOnly) { - return this.getRelative("prev()", "preceding", true, by, index, timeout, eleOnly); - } - - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param index 前面第几个查询结果,0开始 - * @return 兄弟元素 - */ - public T prev(Integer index) { - return prev(index, null); - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param index 前面第几个查询结果,0开始 - * @param timeout 查找节点的超时时间(秒) - * @return 兄弟元素 - */ - public T prev(Integer index, Double timeout) { - return prev(index, timeout, true); - } - - public T prev(Integer index, Double timeout, Boolean eleOnly) { - return this.getRelative("prev()", "preceding", true, index, timeout, eleOnly); - } - - /** - * 返回后面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param by 用于筛选的查询语法 - * @param index 第几个查询结果,1开始 - * @param timeout 查找节点的超时时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 兄弟元素 - */ - @Override - public T next(By by, Integer index, Double timeout, Boolean eleOnly) { - return this.getRelative("next()", "following", true, by, index, timeout, eleOnly); - } - - /** - * 返回后面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param loc 用于筛选的查询语法 - * @param index 第几个查询结果,1开始 - * @param timeout 查找节点的超时时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 兄弟元素 - */ - - @Override - public T next(String loc, Integer index, Double timeout, Boolean eleOnly) { - return this.getRelative("next()", "following", true, loc, index, timeout, eleOnly); - } - - /** - * 返回当前元素后面一个符合条件的同级元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param index 第几个查询结果,0开始 - * @param timeout 查找节点的超时时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 兄弟元素或节点文本 - */ - @Override - public T next(Integer index, Double timeout, Boolean eleOnly) { - return this.getRelative("next()", "following", true, index, timeout, eleOnly); - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param by 用于筛选的查询语法 - * @return 本元素前面的某个元素或节点 - */ - public T before(By by) { - return before(by, null); - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param by 用于筛选的查询语法 - * @param index 前面第几个查询结果,1开始 - * @return 本元素前面的某个元素或节点 - */ - public T before(By by, Integer index) { - return before(by, index, null); - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param by 用于筛选的查询语法 - * @param index 前面第几个查询结果,1开始 - * @param timeout 查找节点的超时时间(秒) - * @return 本元素前面的某个元素或节点 - */ - public T before(By by, Integer index, Double timeout) { - return before(by, index, timeout, true); - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param by 用于筛选的查询语法 - * @param index 前面第几个查询结果,1开始 - * @param timeout 查找节点的超时时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 本元素前面的某个元素或节点 - */ - public T before(By by, Integer index, Double timeout, Boolean eleOnly) { - return this.getRelative("before()", "preceding", false, by, index, timeout, eleOnly); - } - - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @return 本元素前面的某个元素或节点 - */ - public T before() { - return before(""); - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param loc 用于筛选的查询语法 - * @return 本元素前面的某个元素或节点 - */ - public T before(String loc) { - return before(loc.isEmpty() ? null : loc, null); - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param loc 用于筛选的查询语法 - * @param index 前面第几个查询结果,1开始 - * @return 本元素前面的某个元素或节点 - */ - public T before(String loc, Integer index) { - return before(loc, index, null); - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param loc 用于筛选的查询语法 - * @param index 前面第几个查询结果,1开始 - * @param timeout 查找节点的超时时间(秒) - * @return 本元素前面的某个元素或节点 - */ - public T before(String loc, Integer index, Double timeout) { - return before(loc, index, timeout, true); - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param loc 用于筛选的查询语法 - * @param index 前面第几个查询结果,1开始 - * @param timeout 查找节点的超时时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 本元素前面的某个元素或节点 - */ - public T before(String loc, Integer index, Double timeout, Boolean eleOnly) { - return this.getRelative("before()", "preceding", false, loc, index, timeout, eleOnly); - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param index 前面第几个查询结果,1开始 - * @return 本元素前面的某个元素或节点 - */ - public T before(Integer index) { - return before(index, null); - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param index 前面第几个查询结果,1开始 - * @param timeout 查找节点的超时时间(秒) - * @return 本元素前面的某个元素或节点 - */ - public T before(Integer index, Double timeout) { - return before(index, timeout, true); - } - - /** - * 返回前面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param index 前面第几个查询结果,1开始 - * @param timeout 查找节点的超时时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 本元素前面的某个元素或节点 - */ - public T before(Integer index, Double timeout, Boolean eleOnly) { - return this.getRelative("before()", "preceding", false, index, timeout, eleOnly); - } - - /** - * 返回后面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param by 用于筛选的查询语法 - * @return 本元素后面的某个元素或节点 - */ - public T after(By by) { - return after(by, null); - } - - /** - * 返回后面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param by 用于筛选的查询语法 - * @param index 前面第几个查询结果,1开始 - * @return 本元素后面的某个元素或节点 - */ - public T after(By by, Integer index) { - return after(by, index, null); - } - - /** - * 返回后面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param by 用于筛选的查询语法 - * @param index 前面第几个查询结果,1开始 - * @param timeout 查找节点的超时时间(秒) - * @return 本元素后面的某个元素或节点 - */ - public T after(By by, Integer index, Double timeout) { - return after(by, index, timeout, true); - } - - /** - * 返回后面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param by 用于筛选的查询语法 - * @param index 前面第几个查询结果,1开始 - * @param timeout 查找节点的超时时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 本元素后面的某个元素或节点 - */ - public T after(By by, Integer index, Double timeout, Boolean eleOnly) { - return this.getRelative("after()", "following", false, by, index, timeout, eleOnly); - } - - /** - * 返回后面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @return 本元素后面的某个元素或节点 - */ - public T after() { - return after(""); - } - - /** - * 返回后面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param loc 用于筛选的查询语法 - * @return 本元素后面的某个元素或节点 - */ - public T after(String loc) { - return after(loc.isEmpty() ? null : loc, null); - } - - /** - * 返回后面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param loc 用于筛选的查询语法 - * @param index 前面第几个查询结果,1开始 - * @return 本元素后面的某个元素或节点 - */ - public T after(String loc, Integer index) { - return after(loc, index, null); - } - - /** - * 返回后面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param loc 用于筛选的查询语法 - * @param index 前面第几个查询结果,1开始 - * @param timeout 查找节点的超时时间(秒) - * @return 本元素后面的某个元素或节点 - */ - public T after(String loc, Integer index, Double timeout) { - return after(loc, index, timeout, true); - } - - /** - * 返回后面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param loc 用于筛选的查询语法 - * @param index 前面第几个查询结果,1开始 - * @param timeout 查找节点的超时时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 本元素后面的某个元素或节点 - */ - public T after(String loc, Integer index, Double timeout, Boolean eleOnly) { - return this.getRelative("after()", "following", false, loc, index, timeout, eleOnly); - } - - /** - * 返回后面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param index 前面第几个查询结果,1开始 - * @return 本元素后面的某个元素或节点 - */ - public T after(Integer index) { - return after(index, null); - } - - /** - * 返回后面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param index 前面第几个查询结果,1开始 - * @param timeout 查找节点的超时时间(秒) - * @return 本元素后面的某个元素或节点 - */ - public T after(Integer index, Double timeout) { - return after(index, timeout, true); - } - - /** - * 返回后面的一个兄弟元素,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param index 前面第几个查询结果,1开始 - * @param timeout 查找节点的超时时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 本元素后面的某个元素或节点 - */ - public T after(Integer index, Double timeout, Boolean eleOnly) { - return this.getRelative("after()", "following", false, index, timeout, eleOnly); - } - - /** - * 返回直接子元素元素或节点组成的列表,可用查询语法筛选 - * - * @param by 用于筛选的查询语法 - * @return 本元素后面的某个元素或节点 - */ - public List children(By by) { - return children(by, null); - } - - /** - * 返回直接子元素元素或节点组成的列表,可用查询语法筛选 - * - * @param by 用于筛选的查询语法 - * @param timeout 查找节点的超时时间(秒) - * @return 本元素后面的某个元素或节点 - */ - public List children(By by, Double timeout) { - return children(by, timeout, true); - } - - /** - * 返回直接子元素元素或节点组成的列表,可用查询语法筛选 - * - * @param by 用于筛选的查询语法 - * @param timeout 查找节点的超时时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 本元素后面的某个元素或节点 - */ - public List children(By by, Double timeout, Boolean eleOnly) { - String loc; - if (by == null) { - loc = eleOnly ? "*" : "node()"; - } else { - by = Locator.getLoc(by, true, false); - if (by.getName().equals(BySelect.CSS_SELECTOR)) - throw new IllegalArgumentException("此css selector语法不受支持,请换成xpath。"); - else loc = by.getValue().replaceAll("^[.\\s/]+", ""); - } - loc = "xpath:./" + loc; - return this._ele(loc, timeout, null, true, null, null); - } - - /** - * 返回直接子元素元素或节点组成的列表,可用查询语法筛选 - * - * @return 本元素后面的某个元素或节点 - */ - public List children() { - return children(""); - } - - /** - * 返回直接子元素元素或节点组成的列表,可用查询语法筛选 - * - * @param loc 用于筛选的查询语法 - * @return 本元素后面的某个元素或节点 - */ - public List children(String loc) { - return children(loc.isEmpty() ? null : loc, null); - } - - /** - * 返回直接子元素元素或节点组成的列表,可用查询语法筛选 - * - * @param loc 用于筛选的查询语法 - * @param timeout 查找节点的超时时间(秒) - * @return 本元素后面的某个元素或节点 - */ - public List children(String loc, Double timeout) { - return children(loc, timeout, true); - } - - /** - * 返回直接子元素元素或节点组成的列表,可用查询语法筛选 - * - * @param loc 用于筛选的查询语法 - * @param timeout 查找节点的超时时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 本元素后面的某个元素或节点 - */ - public List children(String loc, Double timeout, Boolean eleOnly) { - if (loc == null || loc.isEmpty()) { - loc = eleOnly ? "*" : "node()"; - } else { - By by = Locator.getLoc(loc, true, false); - if (by.getName().equals(BySelect.CSS_SELECTOR)) - throw new IllegalArgumentException("此css selector语法不受支持,请换成xpath。"); - else { - loc = by.getValue().replaceAll("^[.\\s/]+", ""); - } - } - loc = "xpath:./" + loc; - return this._ele(loc, timeout, null, true, null, null); - } - - - /** - * 获取前面符合条件的同级列表 - * - * @return 同级元素或节点文本组成的列表 - */ - public List prevs() { - return prevs("", null); - } - - /** - * 获取前面符合条件的同级列表 - * - * @param loc 查询元素 - * @return 同级元素或节点文本组成的列表 - */ - public List prevs(String loc) { - return prevs(loc.isEmpty() ? null : loc, null); - } - - /** - * 获取前面符合条件的同级列表 - * - * @param loc 查询元素 - * @param timeout 等待时间 - * @return 同级元素或节点文本组成的列表 - */ - public List prevs(String loc, Double timeout) { - return prevs(loc, timeout, true); - } - - /** - * 回前面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param loc 查询元素 - * @param timeout 等待时间 - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 兄弟元素的列表 - */ - public List prevs(String loc, Double timeout, Boolean eleOnly) { - return this.getRelatives(null, loc, "preceding", true, timeout, eleOnly); - } - - /** - * 获取前面符合条件的同级列表 - * - * @param by 查询元素 - * @return 同级元素或节点文本组成的列表 - */ - public List prevs(By by) { - return prevs(by, null); - } - - /** - * 获取前面符合条件的同级列表 - * - * @param by 查询元素 - * @param timeout 等待时间 - * @return 同级元素或节点文本组成的列表 - */ - public List prevs(By by, Double timeout) { - return prevs(by, timeout, true); - } - - - /** - * 回前面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param by 查询元素 - * @param timeout 等待时间 - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 兄弟元素的列表 - */ - public List prevs(By by, Double timeout, Boolean eleOnly) { - return this.getRelatives(null, by, "preceding", true, timeout, eleOnly); - } - - /** - * 返回后面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param by 用于筛选的查询语法 - * @return 兄弟元素或节点文本组成的列表 - */ - public List nexts(By by) { - return nexts(by, null); - } - - /** - * 返回后面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param by 用于筛选的查询语法 - * @param timeout 查找节点的超时时间 - * @return 兄弟元素或节点文本组成的列表 - */ - public List nexts(By by, Double timeout) { - return nexts(by, timeout, true); - } - - /** - * 返回后面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param by 用于筛选的查询语法 - * @param timeout 查找节点的超时时间 - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 兄弟元素或节点文本组成的列表 - */ - public List nexts(By by, Double timeout, Boolean eleOnly) { - return this.getRelatives(null, by, "following", true, timeout, eleOnly); - } - - /** - * 返回后面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @return 兄弟元素或节点文本组成的列表 - */ - - public List nexts() { - return nexts(""); - } - - /** - * 返回后面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param loc 用于筛选的查询语法 - * @return 兄弟元素或节点文本组成的列表 - */ - public List nexts(String loc) { - return nexts(loc.isEmpty() ? null : loc, null); - } - - /** - * 返回后面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param loc 用于筛选的查询语法 - * @param timeout 查找节点的超时时间 - * @return 兄弟元素或节点文本组成的列表 - */ - public List nexts(String loc, Double timeout) { - return nexts(loc, timeout, true); - } - - /** - * 返回后面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param loc 用于筛选的查询语法 - * @param timeout 查找节点的超时时间 - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 兄弟元素或节点文本组成的列表 - */ - public List nexts(String loc, Double timeout, Boolean eleOnly) { - return this.getRelatives(null, loc, "following", true, timeout, eleOnly); - } - - /** - * 返回后面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param by 用于筛选的查询语法 - * @return 本元素前面的元素或节点组成的列表 - */ - public List befores(By by) { - return this.befores(by, null); - } - - /** - * 返回后面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param by 用于筛选的查询语法 - * @param timeout 查找节点的超时时间(秒) - * @return 本元素前面的元素或节点组成的列表 - */ - public List befores(By by, Double timeout) { - return this.befores(by, timeout, true); - } - - /** - * 返回后面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param by 用于筛选的查询语法 - * @param timeout 查找节点的超时时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 本元素前面的元素或节点组成的列表 - */ - public List befores(By by, Double timeout, Boolean eleOnly) { - return this.getRelatives(null, by, "preceding", false, timeout, eleOnly); - } - - /** - * 返回后面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @return 本元素前面的元素或节点组成的列表 - */ - public List befores() { - return this.befores(""); - } - - /** - * 返回后面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param loc 用于筛选的查询语法 - * @return 本元素前面的元素或节点组成的列表 - */ - public List befores(String loc) { - return this.befores(loc.isEmpty() ? null : loc, null); - } - - /** - * 返回后面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param loc 用于筛选的查询语法 - * @param timeout 查找节点的超时时间(秒) - * @return 本元素前面的元素或节点组成的列表 - */ - public List befores(String loc, Double timeout) { - return this.befores(loc, timeout, true); - } - - /** - * 返回后面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param loc 用于筛选的查询语法 - * @param timeout 查找节点的超时时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 本元素前面的元素或节点组成的列表 - */ - public List befores(String loc, Double timeout, Boolean eleOnly) { - return this.getRelatives(null, loc, "preceding", false, timeout, eleOnly); - } - - /** - * 返回前面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param by 用于筛选的查询语法 - * @return 本元素前面的元素或节点组成的列表 - */ - public List afters(By by) { - return this.afters(by, null); - } - - - /** - * 返回前面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param by 用于筛选的查询语法 - * @param timeout 查找节点的超时时间(秒) - * @return 本元素前面的元素或节点组成的列表 - */ - public List afters(By by, Double timeout) { - return this.afters(by, timeout, true); - } - - /** - * 返回前面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param by 用于筛选的查询语法 - * @param timeout 查找节点的超时时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 本元素前面的元素或节点组成的列表 - */ - public List afters(By by, Double timeout, Boolean eleOnly) { - return this.getRelatives(null, by, "following", false, timeout, eleOnly); - } - - - /** - * 返回前面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @return 本元素前面的元素或节点组成的列表 - */ - public List afters() { - return this.afters(""); - } - - /** - * 返回前面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param loc 用于筛选的查询语法 - * @return 本元素前面的元素或节点组成的列表 - */ - public List afters(String loc) { - return this.afters(loc.isEmpty() ? null : loc, null); - } - - /** - * 返回前面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param loc 用于筛选的查询语法 - * @param timeout 查找节点的超时时间(秒) - * @return 本元素前面的元素或节点组成的列表 - */ - public List afters(String loc, Double timeout) { - return this.afters(loc, timeout, true); - } - - /** - * 返回前面全部兄弟元素或节点组成的列表,可用查询语法筛选 - * - * @param loc 用于筛选的查询语法 - * @param timeout 查找节点的超时时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 本元素前面的元素或节点组成的列表 - */ - public List afters(String loc, Double timeout, Boolean eleOnly) { - return this.getRelatives(null, loc, "following", false, timeout, eleOnly); - } - - /** - * *获取一个亲戚元素或节点,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param func: 方法名称 - * @param direction: 方向,'following' 或 'preceding' - * @param brother 查找范围,在同级查找还是整个dom前后查找 - * @param loc: 用于筛选的查询语法 - * @param timeout: 查找节点的超时时间(秒) - * @param eleOnly: 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 本元素前面的某个元素或节点 - */ - private T getRelative(String func, String direction, Boolean brother, Integer loc, Double timeout, Boolean eleOnly) { - List relatives = this.getRelatives(loc, "", direction, brother, timeout, eleOnly); - if (!relatives.isEmpty()) return relatives.get(0); - if (Settings.raiseWhenEleNotFound) - throw new ElementNotFoundError(func, Map.of("loc", "", "index", loc, "eleOnly", eleOnly)); - return null; - } - - /** - * *获取一个亲戚元素或节点,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param func: 方法名称 - * @param direction: 方向,'following' 或 'preceding' - * @param brother 查找范围,在同级查找还是整个dom前后查找 - * @param loc: 用于筛选的查询语法 - * @param index: 前面第几个查询结果,1开始 - * @param timeout: 查找节点的超时时间(秒) - * @param eleOnly: 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 本元素前面的某个元素或节点 - */ - private T getRelative(String func, String direction, Boolean brother, String loc, Integer index, Double timeout, Boolean eleOnly) { - List relatives = this.getRelatives(index, loc, direction, brother, timeout, eleOnly); - if (!relatives.isEmpty()) return relatives.get(0); - if (Settings.raiseWhenEleNotFound) - throw new ElementNotFoundError(func, Map.of("loc", "", "index", index, "eleOnly", eleOnly)); - return null; - } - - /** - * *获取一个亲戚元素或节点,可用查询语法筛选,可指定返回筛选结果的第几个 - * - * @param func: 方法名称 - * @param direction: 方向,'following' 或 'preceding' - * @param brother 查找范围,在同级查找还是整个dom前后查找 - * @param by: 用于筛选的查询语法 - * @param index: 前面第几个查询结果,1开始 - * @param timeout: 查找节点的超时时间(秒) - * @param eleOnly: 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 本元素前面的某个元素或节点 - */ - private T getRelative(String func, String direction, Boolean brother, By by, Integer index, Double timeout, Boolean eleOnly) { - List relatives = this.getRelatives(index, by, direction, brother, timeout, eleOnly); - if (!relatives.isEmpty()) return relatives.get(0); - if (Settings.raiseWhenEleNotFound) - throw new ElementNotFoundError(func, Map.of("loc", "", "index", index, "eleOnly", eleOnly)); - return null; - } - - /** - * @param index 获取第几个,该参数不为null时只获取该编号的元素 - * @param loc 用于筛选的查询语法 - * @param direction 'following' 或 'preceding',查找的方向 - * @param brother 查找范围,在同级查找还是整个dom前后查找 - * @param timeout 查找等待时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 元素对象或字符串 - */ - private List getRelatives(Integer index, String loc, String direction, Boolean brother, Double timeout, Boolean eleOnly) { - //获取查到范围 - String brotherStr = brother ? "-sibling" : ""; - if (loc == null || loc.isEmpty()) { - loc = eleOnly ? "*" : "node()"; - } else { - By by = Locator.getLoc(loc, true, false); - if (!by.getName().equals(BySelect.CSS_SELECTOR)) - throw new IllegalArgumentException("此css selector语法不受支持,请换成xpath。"); - else loc = by.getValue().replaceAll("^[.\\s/]+", ""); - } - loc = "xpath:./" + direction + brotherStr + "::" + loc; - if (index != null) index = "following".equals(direction) ? index : -index; - return this._ele(loc, timeout, index, true, false, null); - } - - /** - * @param index 获取第几个,该参数不为null时只获取该编号的元素 - * @param by 用于筛选的查询语法 - * @param direction 'following' 或 'preceding',查找的方向 - * @param brother 查找范围,在同级查找还是整个dom前后查找 - * @param timeout 查找等待时间(秒) - * @param eleOnly 是否只获取元素,为False时把文本、注释节点也纳入 - * @return 元素对象或字符串 - */ - private List getRelatives(Integer index, By by, String direction, Boolean brother, Double timeout, Boolean eleOnly) { - //获取查到范围 - String brotherStr = brother ? "-sibling" : ""; - String loc; - if (by == null) { - loc = eleOnly ? "*" : "node()"; - } else { - by = Locator.getLoc(by, true, false); - if (!by.getName().equals(BySelect.CSS_SELECTOR)) - throw new IllegalArgumentException("此css selector语法不受支持,请换成xpath。"); - else loc = by.getValue().replaceAll("^[.\\s/]+", ""); - } - loc = "xpath:./" + direction + brotherStr + "::" + loc; - if (index != null) index = "following".equals(direction) ? index : -index; - return this._ele(loc, timeout, index, true, false, null); - } - - //---------------------------------------------- - - /** - * @return 返回元素所有属性及值 - */ - public abstract Map attrs(); - - /** - * 返回处理的元素内文本 - */ - public abstract String text(); - - /** - * 返回未格式化处理的元素内文本 - */ - public abstract String rawText(); - - /** - * 返回attribute属性值 - * - * @param attr 属性名 - * @return 属性值文本,没有该属性返回null - */ - public abstract String attr(String attr); - - /** - * 获取css路径或xpath路径 - * - * @param mode 'css' 或 'xpath' - * @return css路径或xpath路径 - */ - protected abstract String getElePath(ElePathMode mode); -} diff --git a/java/src/main/java/com/ll/DrissonPage/base/Driver.java b/java/src/main/java/com/ll/DrissonPage/base/Driver.java deleted file mode 100644 index fc53df3..0000000 --- a/java/src/main/java/com/ll/DrissonPage/base/Driver.java +++ /dev/null @@ -1,450 +0,0 @@ -package com.ll.DrissonPage.base; - -import com.alibaba.fastjson.JSON; -import com.alibaba.fastjson.JSONObject; -import com.ll.DrissonPage.functions.Settings; -import lombok.Getter; -import lombok.Setter; -import okhttp3.*; -import okio.ByteString; -import org.jetbrains.annotations.NotNull; - -import java.util.ArrayList; -import java.util.HashMap; -import java.util.Map; -import java.util.concurrent.BlockingQueue; -import java.util.concurrent.ConcurrentHashMap; -import java.util.concurrent.LinkedBlockingQueue; -import java.util.concurrent.TimeUnit; -import java.util.concurrent.atomic.AtomicBoolean; -import java.util.concurrent.atomic.AtomicInteger; -import java.util.concurrent.atomic.AtomicReference; - -/** - * 驱动 okHTTP框架 - * - * @author 陆 - * @address click - * @original DrissionPage - */ - -public class Driver { - /** - * 标签id - */ - @Getter - private final String id; - /** - * 浏览器连接地址 - */ - @Getter - private final String address; - /** - * 标签页类型 - */ - @Getter - private final String type; - @Setter - @Getter - private boolean debug; - private final String websocketUrl; - private final AtomicInteger curId; - /** - * 会话返回值 - */ - private final AtomicReference webSocketMsg = new AtomicReference<>(null); - private final Thread recvThread; - private final Thread handleEventThread; - @Getter - private AtomicBoolean stopped; - private final BlockingQueue> eventQueue; - private final BlockingQueue> immediateEventQueue; - private final ConcurrentHashMap eventHandlers; - private final ConcurrentHashMap immediateEventHandlers; - private final ConcurrentHashMap>> methodResults; - /** - * 创建这个驱动的对象 - */ - @Getter - @Setter - private Occupant occupant; - private boolean alertFlag; - private static final OkHttpClient okHttpClient = new OkHttpClient().newBuilder() - .connectTimeout(5, TimeUnit.SECONDS) - .callTimeout(5, TimeUnit.SECONDS) - .readTimeout(5, TimeUnit.SECONDS) - .pingInterval(5, TimeUnit.SECONDS).build(); - /** - * 会话驱动 - */ - private WebSocket ws; - private Thread handleImmediateEventThread; - - public Driver(String tabId, String tabType, String address) { - this(tabId, tabType, address, null); - } - - /** - * 驱动 - * - * @param tabId 标签id - * @param tabType 标签页类型 - * @param address 浏览器连接地址 - */ - public Driver(String tabId, String tabType, String address, Occupant occupant) { - this.id = tabId; - this.address = address; - this.type = tabType; - this.occupant = occupant; - this.debug = false; - this.alertFlag = false; - this.websocketUrl = "ws://" + address + "/devtools/" + tabType + "/" + tabId; - this.curId = new AtomicInteger(0); - this.ws = null; - - - this.recvThread = new Thread(this::recvLoop); - this.handleEventThread = new Thread(this::handleEventLoop); - this.recvThread.setDaemon(true); - this.handleEventThread.setDaemon(true); - this.handleImmediateEventThread = null; - - this.stopped = new AtomicBoolean(); - - this.eventHandlers = new ConcurrentHashMap<>(); - this.immediateEventHandlers = new ConcurrentHashMap<>(); - this.methodResults = new ConcurrentHashMap<>(); - this.eventQueue = new LinkedBlockingQueue<>(); - this.immediateEventQueue = new LinkedBlockingQueue<>(); - start(); - } - - /** - * 发送信息到浏览器,并返回浏览器返回的信息 - * - * @param message 发送给浏览器的数据 - * @param timeout 超时时间,为null表示无时间 - * @return 浏览器返回的数据 - */ - private JSONObject send(Map message, double timeout) { - message = new HashMap<>(message); - int wsId = curId.incrementAndGet(); - message.put("id", wsId); - String messageJson = JSON.toJSONString(message); - - if (this.debug) System.out.println("发->" + messageJson); - //计算等待时间 - long endTime = (long) (System.currentTimeMillis() + timeout * 1000L); - LinkedBlockingQueue> value = new LinkedBlockingQueue<>(); - methodResults.put(wsId, value); - try { - ws.send(messageJson); - if (timeout == 0) { - methodResults.remove(wsId); - return new JSONObject(Map.of("id", wsId, "result", Map.of())); - } - } catch (Exception e) { - e.printStackTrace(); - methodResults.remove(wsId); - return new JSONObject(Map.of("error", Map.of("message", "connection disconnected"), "type", "connection_error")); - } - int i = 5; - long endTimes = System.currentTimeMillis() + 1000L; - while (!stopped.get()) { - try { - Map result = methodResults.get(wsId).poll(5, TimeUnit.SECONDS); - if (result == null && System.currentTimeMillis() < endTimes) continue; - if (result == null && i > 0 && System.currentTimeMillis() > endTimes) { - i--; - endTimes = System.currentTimeMillis() + 1000L; - if (debug) System.out.println("超时或者丢包,重新发送:->" + messageJson); - ws.send(messageJson); - continue; - } - methodResults.remove(wsId); - if (result == null) throw new NullPointerException(); - return new JSONObject(result); - } catch (InterruptedException | NullPointerException | IllegalArgumentException e) { -// e.printStackTrace(); - String string = message.get("method").toString(); - if (alertFlag && string.startsWith("Input.") || string.startsWith("Runtime.")) { - return new JSONObject(Map.of("error", Map.of("message", "alert exists."), "type", "alert_exists")); - } - if (timeout > 0 && System.currentTimeMillis() > endTime) { - methodResults.remove(wsId); - return alertFlag ? new JSONObject(Map.of("error", Map.of("message", "alert exists."), "type", "alert_exists")) : new JSONObject(Map.of("error", Map.of("message", "timeout"), "type", "timeout")); - } - } - } - - return new JSONObject(Map.of("error", Map.of("message", "connection disconnected"), "type", "connection_error")); - } - - /** - * 接收浏览器信息的守护线程方法 - */ - private void recvLoop() { - while (!stopped.get()) { - JSONObject msg; - try { - String andSet = webSocketMsg.getAndSet(null); - if (andSet != null) { - msg = JSONObject.parseObject(andSet); - } else continue; - } catch (Exception e) { - if (stop()) return; - return; - - } - if (this.debug) System.out.println("<-收" + msg); - - if (msg.containsKey("method")) { - if (msg.getString("method").startsWith("Page.javascriptDialog")) { - alertFlag = msg.getString("method").endsWith("Opening"); - } - MyRunnable function = immediateEventHandlers.get(msg.getString("method")); - if (function != null) { - this.handleImmediateEvent(function, msg.getOrDefault("params", new HashMap<>())); - } else { - try { - eventQueue.put(msg); - } catch (InterruptedException e) { - throw new RuntimeException(e); - } - } - } else { - int i = 1000; - Integer integer = msg.getInteger("id"); - while (i-- > 0 && integer != null && !methodResults.containsKey(integer)) { - try { - Thread.sleep(10); - } catch (InterruptedException e) { - throw new RuntimeException(e); - } - } - if (methodResults.containsKey(integer)) { - try { - methodResults.get(integer).put(msg); - } catch (InterruptedException e) { - throw new RuntimeException(e); - } - } else if (this.debug) { - System.out.println("未知错误->" + msg); - - } - } - - } - } - - /** - * 当接收到浏览器信息,执行已绑定的方法 - */ - private void handleEventLoop() { - while (!stopped.get()) { - Map event; - try { - event = eventQueue.poll(2, TimeUnit.SECONDS); - } catch (InterruptedException e) { - continue; - } - - if (event != null) { - MyRunnable function = eventHandlers.get(event.get("method").toString()); - if (function != null) { - function.setMessage(event.get("params")); - function.run(); - } - } - this.eventQueue.poll(); - - } - } - - private void handleImmediateEventLoop() { - while (!stopped.get() && !immediateEventQueue.isEmpty()) { - Map event; - try { - event = immediateEventQueue.poll(2, TimeUnit.SECONDS); - } catch (InterruptedException e) { - e.printStackTrace(); - continue; - } - if (event != null) { - MyRunnable function = immediateEventHandlers.get(event.get("method").toString()); - if (function != null) { - function.setMessage(event.get("params")); - function.run(); - } - } - - } - } - - /** - * 处理立即执行的动作 - * - * @param function 要运行下方法 - * @param params 方法参数 - */ - private void handleImmediateEvent(MyRunnable function, Object params) { - Map func = new HashMap<>(); - func.put("method", function); - func.put("params", params); - immediateEventQueue.add(func); - - if (handleImmediateEventThread == null || !handleImmediateEventThread.isAlive()) { - handleImmediateEventThread = new Thread(this::handleImmediateEventLoop); - handleImmediateEventThread.setDaemon(true); - handleImmediateEventThread.start(); - } - } - - /** - * 执行cdp方法 - * - * @param method 方法 - * @return 执行结果 - */ - public Object run(String method) { - return run(method, new HashMap<>()); - } - - /** - * 执行cdp方法 - * - * @param method 方法 - * @param params 参数 - * @return 执行结果 - */ - public Object run(String method, Map params) { - if (stopped.get()) return Map.of("error", "connection disconnected", "type", "connection_error"); - params = new HashMap<>(params); - Object timeout1 = params.remove("_timeout"); - double timeout = timeout1 != null ? Float.parseFloat(timeout1.toString()) : Settings.cdpTimeout; - - JSONObject result = this.send(Map.of("method", method, "params", params), timeout); - if (!result.containsKey("result") && result.containsKey("error")) { - HashMap map = new HashMap<>(); - map.put("error", result.getJSONObject("error").get("message")); - map.put("type", result.getOrDefault("type", "call_method_error")); - map.put("method", method); - map.put("args", params); - map.put("timeout", timeout); - return JSON.toJSONString(map); - } else { - return JSON.toJSONString(result.get("result")); - } - } - - - /** - * 启动连接 - */ - private void start() { - this.stopped.set(false); - try { - Request build = new Request(new HttpUrl("socket", "", "", "", 80, new ArrayList<>(), null, null, this.websocketUrl), "GET", Headers.of(), null, new HashMap<>()).newBuilder().url(this.websocketUrl).build(); - ws = okHttpClient.newWebSocket(build, new WebSocketListener() { - @Override - public void onClosed(@NotNull WebSocket webSocket, int code, @NotNull String reason) { - // 关闭事件处理 - stop(); - super.onClosed(webSocket, code, reason); - } - - @Override - public void onClosing(@NotNull WebSocket webSocket, int code, @NotNull String reason) { - super.onClosing(webSocket, code, reason); - } - - @Override - public void onFailure(@NotNull WebSocket webSocket, @NotNull Throwable t, Response response) { - super.onFailure(webSocket, t, response); - } - - @Override - public void onMessage(@NotNull WebSocket webSocket, @NotNull String text) { - webSocketMsg.set(text); - super.onMessage(webSocket, text); - } - - @Override - public void onMessage(@NotNull WebSocket webSocket, @NotNull ByteString bytes) { - super.onMessage(webSocket, bytes); - } - - @Override - public void onOpen(@NotNull WebSocket webSocket, @NotNull Response response) { - super.onOpen(webSocket, response); - } - - }); - recvThread.start(); - handleEventThread.start(); - } catch (Exception e) { - e.printStackTrace(); - stop(); - } - } - - /** - * 中断连接 - */ - public boolean stop() { - stop1(); - while (this.recvThread.isAlive() || this.handleEventThread.isAlive()) { - try { - Thread.sleep(100); - } catch (InterruptedException e) { - throw new RuntimeException(e); - } - } - return true; - } - - /** - * 中断连接 - */ - private void stop1() { - if (debug) System.out.println("关闭"); - if (stopped.get()) return; - stopped.set(!stopped.get()); - if (ws != null) { - ws.close(1000, ""); - ws = null; - } - try { - while (!eventQueue.isEmpty()) { - Map event = eventQueue.poll(); - MyRunnable method = eventHandlers.get(event.get("method").toString()); - if (method != null) { - method.setMessage(event.get("params")); - method.run(); - } - } - } catch (Exception ignored) { - } - eventHandlers.clear(); - methodResults.clear(); - eventQueue.clear(); - if (occupant != null) occupant.onDisconnect(); - } - - public void setCallback(String event, MyRunnable callback) { - setCallback(event, callback, false); - } - - /** - * 绑定cdp event和回调方法 - * - * @param event 方法名称 - * @param callback 绑定到cdp event的回调方法 - * @param immediate 是否要立即处理的动作 - */ - public void setCallback(String event, MyRunnable callback, boolean immediate) { - Map handler = immediate ? immediateEventHandlers : eventHandlers; - if (callback != null) handler.put(event, callback); - else handler.remove(event); - } -} \ No newline at end of file diff --git a/java/src/main/java/com/ll/DrissonPage/base/Driver2.java b/java/src/main/java/com/ll/DrissonPage/base/Driver2.java deleted file mode 100644 index 8d2230e..0000000 --- a/java/src/main/java/com/ll/DrissonPage/base/Driver2.java +++ /dev/null @@ -1,423 +0,0 @@ -package com.ll.DrissonPage.base; - -/** - * 驱动 - * - * @author 陆 - * @address click - * @original DrissionPage - */ - -public class Driver2 { -// /** -// * 标签id -// */ -// @Getter -// private final String id; -// /** -// * 浏览器连接地址 -// */ -// @Getter -// private final String address; -// /** -// * 标签页类型 -// */ -// @Getter -// private final String type; -// private final boolean debug; -// private final String websocketUrl; -// private final AtomicInteger curId; -// /** -// * 会话返回值 -// */ -// private final AtomicReference webSocketMsg = new AtomicReference<>(null); -// private final Thread recvThread; -// private final Thread handleEventThread; -// @Getter -// -// private final AtomicBoolean stopped; -// private final BlockingQueue> eventQueue; -// private final BlockingQueue> immediateEventQueue; -// private final Map eventHandlers; -// private final Map immediateEventHandlers; -// private final Map>> methodResults; -// /** -// * 创建这个驱动的对象 -// */ -// @Getter -// @Setter -// private Occupant occupant; -// private boolean alertFlag; -// /** -// * 会话驱动 -// */ -// private WebSocket ws; -// private Thread handleImmediateEventThread; -// -// public Driver2(String tabId, String tabType, String address) { -// this(tabId, tabType, address, null); -// } -// -// /** -// * 驱动 -// * -// * @param tabId 标签id -// * @param tabType 标签页类型 -// * @param address 浏览器连接地址 -// */ -// public Driver2(String tabId, String tabType, String address, Occupant occupant) { -// this.id = tabId; -// this.address = address; -// this.type = tabType; -// this.occupant = occupant; -// this.debug = true; -// this.alertFlag = false; -// this.websocketUrl = "ws://" + address + "/devtools/" + tabType + "/" + tabId; -// this.curId = new AtomicInteger(0); -// this.ws = null; -// -// -// this.recvThread = new Thread(this::recvLoop); -// this.handleEventThread = new Thread(this::handleEventLoop); -// this.recvThread.setDaemon(true); -// this.handleEventThread.setDaemon(true); -// this.handleImmediateEventThread = null; -// -// this.stopped = new AtomicBoolean(); -// -// this.eventHandlers = new ConcurrentHashMap<>(); -// this.immediateEventHandlers = new ConcurrentHashMap<>(); -// this.methodResults = new ConcurrentHashMap<>(); -// this.eventQueue = new LinkedBlockingQueue<>(); -// this.immediateEventQueue = new LinkedBlockingQueue<>(); -// start(); -// } -// -// /** -// * 发送信息到浏览器,并返回浏览器返回的信息 -// * -// * @param message 发送给浏览器的数据 -// * @param timeout 超时时间,为null表示无时间 -// * @return 浏览器返回的数据 -// */ -// private JSONObject send(Map message, double timeout) { -// message = new HashMap<>(message); -// int wsId = curId.incrementAndGet(); -// message.put("id", wsId); -// String messageJson = JSON.toJSONString(message); -// -// if (this.debug) System.out.println("发->" + messageJson); -// //计算等待时间 -// long endTime = (long) (System.currentTimeMillis() + timeout * 1000L); -// LinkedBlockingQueue> value = new LinkedBlockingQueue<>(); -// methodResults.put(wsId, value); -// try { -// ws.send(messageJson); -// if (timeout == 0) { -// methodResults.remove(wsId); -// return new JSONObject(Map.of("id", wsId, "result", Map.of())); -// } -// } catch (WebsocketNotConnectedException e) { -// e.printStackTrace(); -// methodResults.remove(wsId); -// return new JSONObject(Map.of("error", Map.of("message", "connection disconnected"), "type", "connection_error")); -// } -// int i = 5; -// long endTimes = System.currentTimeMillis() + 1000L; -// while (!stopped.get()) { -// try { -// Map result = methodResults.get(wsId).poll(200, TimeUnit.MILLISECONDS); -// if (result == null && System.currentTimeMillis() < endTimes) continue; -// if (result == null && i > 0 && System.currentTimeMillis() > endTimes) { -// i--; -// endTimes = System.currentTimeMillis() + 1000L; -// ws.send(messageJson); -// continue; -// } -// methodResults.remove(wsId); -// if (result == null) throw new NullPointerException(); -// return new JSONObject(result); -// } catch (InterruptedException | NullPointerException | IllegalArgumentException e) { -//// e.printStackTrace(); -// String string = message.get("method").toString(); -// if (alertFlag && string.startsWith("Input.") || string.startsWith("Runtime.")) { -// return new JSONObject(Map.of("error", Map.of("message", "alert exists."), "type", "alert_exists")); -// } -// if (timeout > 0 && System.currentTimeMillis() > endTime) { -// methodResults.remove(wsId); -// return alertFlag ? new JSONObject(Map.of("error", Map.of("message", "alert exists."), "type", "alert_exists")) : new JSONObject(Map.of("error", Map.of("message", "timeout"), "type", "timeout")); -// } -// } -// } -// -// return new JSONObject(Map.of("error", Map.of("message", "connection disconnected"), "type", "connection_error")); -// } -// -// /** -// * 接收浏览器信息的守护线程方法 -// */ -// private void recvLoop() { -// while (!stopped.get()) { -// JSONObject msg; -// try { -// String andSet = webSocketMsg.getAndSet(null); -// if (andSet != null) { -// msg = JSONObject.parseObject(andSet); -// } else continue; -// } catch (Exception e) { -// if (stop()) return; -// return; -// -// } -// if (this.debug) System.out.println("<-收" + msg); -// -// if (msg.containsKey("method")) { -// if (msg.getString("method").startsWith("Page.javascriptDialog")) { -// alertFlag = msg.getString("method").endsWith("Opening"); -// } -// MyRunnable function = immediateEventHandlers.get(msg.getString("method")); -// if (function != null) { -// this.handleImmediateEvent(function, msg.getOrDefault("params", new HashMap<>())); -// } else { -// try { -// eventQueue.put(msg); -// } catch (InterruptedException e) { -// throw new RuntimeException(e); -// } -// } -// } else { -// int i = 1000; -// Integer integer = msg.getInteger("id"); -// while (i-- > 0 && integer != null && !methodResults.containsKey(integer)) { -// try { -// Thread.sleep(10); -// } catch (InterruptedException e) { -// throw new RuntimeException(e); -// } -// } -// if (methodResults.containsKey(integer)) { -// try { -// methodResults.get(integer).put(msg); -// } catch (InterruptedException e) { -// throw new RuntimeException(e); -// } -// } else if (this.debug) { -// System.out.println("未知错误->" + msg); -// -// } -// } -// -// } -// } -// -// /** -// * 当接收到浏览器信息,执行已绑定的方法 -// */ -// private void handleEventLoop() { -// while (!stopped.get()) { -// Map event; -// try { -// event = eventQueue.poll(1, TimeUnit.SECONDS); -// } catch (InterruptedException e) { -// continue; -// } -// -// if (event != null) { -// MyRunnable function = eventHandlers.get(event.get("method").toString()); -// if (function != null) { -// function.setMessage(event.get("params")); -// function.run(); -// } -// } -// this.eventQueue.poll(); -// -// } -// } -// -// private void handleImmediateEventLoop() { -// while (!stopped.get() && !immediateEventQueue.isEmpty()) { -// Map event; -// try { -// event = immediateEventQueue.poll(1, TimeUnit.SECONDS); -// } catch (InterruptedException e) { -// e.printStackTrace(); -// continue; -// } -// if (event != null) { -// MyRunnable function = immediateEventHandlers.get(event.get("method").toString()); -// if (function != null) { -// function.setMessage(event.get("params")); -// function.run(); -// } -// } -// -// } -// } -// -// /** -// * 处理立即执行的动作 -// * -// * @param function 要运行下方法 -// * @param params 方法参数 -// */ -// private void handleImmediateEvent(MyRunnable function, Object params) { -// Map func = new HashMap<>(); -// func.put("method", function); -// func.put("params", params); -// immediateEventQueue.add(func); -// -// if (handleImmediateEventThread == null || !handleImmediateEventThread.isAlive()) { -// handleImmediateEventThread = new Thread(this::handleImmediateEventLoop); -// handleImmediateEventThread.setDaemon(true); -// handleImmediateEventThread.start(); -// } -// } -// -// /** -// * 执行cdp方法 -// * -// * @param method 方法 -// * @return 执行结果 -// */ -// public Object run(String method) { -// return run(method, new HashMap<>()); -// } -// -// /** -// * 执行cdp方法 -// * -// * @param method 方法 -// * @param params 参数 -// * @return 执行结果 -// */ -// public Object run(String method, Map params) { -// if (stopped.get()) return Map.of("error", "connection disconnected", "type", "connection_error"); -// params = new HashMap<>(params); -// Object timeout1 = params.remove("_timeout"); -// double timeout = timeout1 != null ? Float.parseFloat(timeout1.toString()) : 30.0; -// -// JSONObject result = this.send(Map.of("method", method, "params", params), timeout); -// if (!result.containsKey("result") && result.containsKey("error")) { -// HashMap map = new HashMap<>(); -// map.put("error", result.getJSONObject("error").get("message")); -// map.put("type", result.getOrDefault("type", "call_method_error")); -// map.put("method", method); -// map.put("args", params); -// map.put("timeout", timeout); -// return JSON.toJSONString(map); -// } else { -// return JSON.toJSONString(result.get("result")); -// } -// } -// -// -// /** -// * 启动连接 -// */ -// private void start() { -// this.stopped.set(false); -// try { -// Request build = new Request(new HttpUrl("socket","","","",80,new ArrayList<>(),null,null,this.websocketUrl), "GET", Headers.of(), null, new HashMap<>()).newBuilder().url(this.websocketUrl).build(); -// OkHttpClient okHttpClient = new OkHttpClient(); -// ws = okHttpClient.newWebSocket(build, new WebSocketListener() { -// @Override -// public void onClosed(@NotNull WebSocket webSocket, int code, @NotNull String reason) { -// // 关闭事件处理 -// stop(); -// super.onClosed(webSocket, code, reason); -// } -// -// @Override -// public void onClosing(@NotNull WebSocket webSocket, int code, @NotNull String reason) { -// super.onClosing(webSocket, code, reason); -// } -// -// @Override -// public void onFailure(@NotNull WebSocket webSocket, @NotNull Throwable t, @Nullable Response response) { -// super.onFailure(webSocket, t, response); -// } -// -// @Override -// public void onMessage(@NotNull WebSocket webSocket, @NotNull String text) { -// webSocketMsg.set(text); -// super.onMessage(webSocket, text); -// } -// -// @Override -// public void onMessage(@NotNull WebSocket webSocket, @NotNull ByteString bytes) { -// super.onMessage(webSocket, bytes); -// } -// -// @Override -// public void onOpen(@NotNull WebSocket webSocket, @NotNull Response response) { -// super.onOpen(webSocket, response); -// } -// -// }); -// recvThread.start(); -// handleEventThread.start(); -// } catch (Exception e) { -// e.printStackTrace(); -// stop(); -// } -// } -// -// /** -// * 中断连接 -// */ -// public boolean stop() { -// stop1(); -// while (this.recvThread.isAlive() || this.handleEventThread.isAlive()) { -// try { -// Thread.sleep(100); -// } catch (InterruptedException e) { -// throw new RuntimeException(e); -// } -// } -// return true; -// } -// -// /** -// * 中断连接 -// */ -// private void stop1() { -// if (stopped.get()) return; -// stopped.set(true); -// if (ws != null) { -// ws.close(1000, ""); -// ws = null; -// } -// try { -// while (!eventQueue.isEmpty()) { -// Map event = eventQueue.poll(); -// MyRunnable method = eventHandlers.get(event.get("method").toString()); -// if (method != null) { -// method.setMessage(event.get("params")); -// method.run(); -// } -// } -// } catch (Exception ignored) { -// } -// eventHandlers.clear(); -// methodResults.clear(); -// eventQueue.clear(); -// if (occupant != null) occupant.onDisconnect(); -// } -// -// public void setCallback(String event, MyRunnable callback) { -// setCallback(event, callback, false); -// } -// -// /** -// * 绑定cdp event和回调方法 -// * -// * @param event 方法名称 -// * @param callback 绑定到cdp event的回调方法 -// * @param immediate 是否要立即处理的动作 -// */ -// public void setCallback(String event, MyRunnable callback, boolean immediate) { -// Map handler = immediate ? immediateEventHandlers : eventHandlers; -// if (callback != null) handler.put(event, callback); -// else handler.remove(event); -// } -} \ No newline at end of file diff --git a/java/src/main/java/com/ll/DrissonPage/base/Driver3.java b/java/src/main/java/com/ll/DrissonPage/base/Driver3.java deleted file mode 100644 index e7b385c..0000000 --- a/java/src/main/java/com/ll/DrissonPage/base/Driver3.java +++ /dev/null @@ -1,426 +0,0 @@ -package com.ll.DrissonPage.base; - -/** - * 驱动 - * - * @author 陆 - * @address click - * @original DrissionPage - */ - -public class Driver3 { -// /** -// * 标签id -// */ -// @Getter -// private final String id; -// /** -// * 浏览器连接地址 -// */ -// @Getter -// private final String address; -// /** -// * 标签页类型 -// */ -// @Getter -// private final String type; -// private final boolean debug; -// private final String websocketUrl; -// private final AtomicInteger curId; -// /** -// * 会话返回值 -// */ -// private final AtomicReference webSocketMsg = new AtomicReference<>(null); -// private final Thread recvThread; -// private final Thread handleEventThread; -// @Getter -// -// private final AtomicBoolean stopped; -// private final BlockingQueue> eventQueue; -// private final BlockingQueue> immediateEventQueue; -// private final Map eventHandlers; -// private final Map immediateEventHandlers; -// private final Map>> methodResults; -// /** -// * 创建这个驱动的对象 -// */ -// @Getter -// @Setter -// private Occupant occupant; -// private boolean alertFlag; -// /** -// * 会话驱动 -// */ -// private WebSocketClient ws; -// private Future webSocketSession; -// private Thread handleImmediateEventThread; -// -// public Driver3(String tabId, String tabType, String address) { -// this(tabId, tabType, address, null); -// } -// -// /** -// * 驱动 -// * -// * @param tabId 标签id -// * @param tabType 标签页类型 -// * @param address 浏览器连接地址 -// */ -// public Driver3(String tabId, String tabType, String address, Occupant occupant) { -// this.id = tabId; -// this.address = address; -// this.type = tabType; -// this.occupant = occupant; -// this.debug = true; -// this.alertFlag = false; -// this.websocketUrl = "ws://" + address + "/devtools/" + tabType + "/" + tabId; -// this.curId = new AtomicInteger(0); -// this.ws = null; -// this.recvThread = new Thread(this::recvLoop); -// this.handleEventThread = new Thread(this::handleEventLoop); -// this.recvThread.setDaemon(true); -// this.handleEventThread.setDaemon(true); -// this.handleImmediateEventThread = null; -// -// this.stopped = new AtomicBoolean(); -// -// this.eventHandlers = new ConcurrentHashMap<>(); -// this.immediateEventHandlers = new ConcurrentHashMap<>(); -// this.methodResults = new ConcurrentHashMap<>(); -// this.eventQueue = new LinkedBlockingQueue<>(); -// this.immediateEventQueue = new LinkedBlockingQueue<>(); -// start(); -// } -// -// /** -// * 发送信息到浏览器,并返回浏览器返回的信息 -// * -// * @param message 发送给浏览器的数据 -// * @param timeout 超时时间,为null表示无时间 -// * @return 浏览器返回的数据 -// */ -// private JSONObject send(Map message, double timeout) { -// message = new HashMap<>(message); -// int wsId = curId.incrementAndGet(); -// message.put("id", wsId); -// String messageJson = JSON.toJSONString(message); -// -// if (this.debug) System.out.println("发->" + messageJson); -// //计算等待时间 -// long endTime = (long) (System.currentTimeMillis() + timeout * 1000L); -// LinkedBlockingQueue> value = new LinkedBlockingQueue<>(); -// methodResults.put(wsId, value); -// try { -// if (ws.isStopping()) return new JSONObject(); -// webSocketSession.get().getRemote().sendString(messageJson); -// if (timeout == 0) { -// methodResults.remove(wsId); -// return new JSONObject(Map.of("id", wsId, "result", Map.of())); -// } -// } catch (WebsocketNotConnectedException | IOException | InterruptedException | ExecutionException e) { -// e.printStackTrace(); -// methodResults.remove(wsId); -// return new JSONObject(Map.of("error", Map.of("message", "connection disconnected"), "type", "connection_error")); -// } -// int i = 5; -// long endTimes = System.currentTimeMillis() + 1000L; -// while (!stopped.get()) { -// try { -// Map result = methodResults.get(wsId).poll(2000, TimeUnit.MILLISECONDS); -// if (result == null && System.currentTimeMillis() < endTimes) continue; -// if (result == null && i > 0 && System.currentTimeMillis() > endTimes) { -// System.out.println("丢包:->" + messageJson); -// i--; -// endTimes = System.currentTimeMillis() + 1000L; -// if (ws.isStopping()) return new JSONObject(); -// webSocketSession.get().getRemote().sendString(messageJson); -// continue; -// } -// methodResults.remove(wsId); -// if (result == null) throw new NullPointerException(); -// return new JSONObject(result); -// } catch (InterruptedException | NullPointerException | IllegalArgumentException | IOException | -// ExecutionException e) { -//// e.printStackTrace(); -// String string = message.get("method").toString(); -// if (alertFlag && string.startsWith("Input.") || string.startsWith("Runtime.")) { -// return new JSONObject(Map.of("error", Map.of("message", "alert exists."), "type", "alert_exists")); -// } -// if (timeout > 0 && System.currentTimeMillis() > endTime) { -// methodResults.remove(wsId); -// return alertFlag ? new JSONObject(Map.of("error", Map.of("message", "alert exists."), "type", "alert_exists")) : new JSONObject(Map.of("error", Map.of("message", "timeout"), "type", "timeout")); -// } -// } -// } -// -// return new JSONObject(Map.of("error", Map.of("message", "connection disconnected"), "type", "connection_error")); -// } -// -// /** -// * 接收浏览器信息的守护线程方法 -// */ -// private void recvLoop() { -// while (!stopped.get()) { -// JSONObject msg; -// try { -// String andSet = webSocketMsg.getAndSet(null); -// if (andSet != null) { -// msg = JSONObject.parseObject(andSet); -// } else continue; -// } catch (Exception e) { -// if (stop()) return; -// return; -// -// } -// if (this.debug) System.out.println("<-收" + msg); -// -// if (msg.containsKey("method")) { -// if (msg.getString("method").startsWith("Page.javascriptDialog")) { -// alertFlag = msg.getString("method").endsWith("Opening"); -// } -// MyRunnable function = immediateEventHandlers.get(msg.getString("method")); -// if (function != null) { -// this.handleImmediateEvent(function, msg.getOrDefault("params", new HashMap<>())); -// } else { -// try { -// eventQueue.put(msg); -// } catch (InterruptedException e) { -// throw new RuntimeException(e); -// } -// } -// } else { -// int i = 1000; -// Integer integer = msg.getInteger("id"); -// while (i-- > 0 && integer != null && !methodResults.containsKey(integer)) { -// try { -// Thread.sleep(10); -// } catch (InterruptedException e) { -// throw new RuntimeException(e); -// } -// } -// if (methodResults.containsKey(integer)) { -// try { -// methodResults.get(integer).put(msg); -// } catch (InterruptedException e) { -// throw new RuntimeException(e); -// } -// } else if (this.debug) { -// System.out.println("未知错误->" + msg); -// -// } -// } -// -// } -// } -// -// /** -// * 当接收到浏览器信息,执行已绑定的方法 -// */ -// private void handleEventLoop() { -// while (!stopped.get()) { -// Map event; -// try { -// event = eventQueue.poll(1, TimeUnit.SECONDS); -// } catch (InterruptedException e) { -// continue; -// } -// -// if (event != null) { -// MyRunnable function = eventHandlers.get(event.get("method").toString()); -// if (function != null) { -// function.setMessage(event.get("params")); -// function.run(); -// } -// } -// this.eventQueue.poll(); -// -// } -// } -// -// private void handleImmediateEventLoop() { -// while (!stopped.get() && !immediateEventQueue.isEmpty()) { -// Map event; -// try { -// event = immediateEventQueue.poll(1, TimeUnit.SECONDS); -// } catch (InterruptedException e) { -// e.printStackTrace(); -// continue; -// } -// if (event != null) { -// MyRunnable function = immediateEventHandlers.get(event.get("method").toString()); -// if (function != null) { -// function.setMessage(event.get("params")); -// function.run(); -// } -// } -// -// } -// } -// -// /** -// * 处理立即执行的动作 -// * -// * @param function 要运行下方法 -// * @param params 方法参数 -// */ -// private void handleImmediateEvent(MyRunnable function, Object params) { -// Map func = new HashMap<>(); -// func.put("method", function); -// func.put("params", params); -// immediateEventQueue.add(func); -// -// if (handleImmediateEventThread == null || !handleImmediateEventThread.isAlive()) { -// handleImmediateEventThread = new Thread(this::handleImmediateEventLoop); -// handleImmediateEventThread.setDaemon(true); -// handleImmediateEventThread.start(); -// } -// } -// -// /** -// * 执行cdp方法 -// * -// * @param method 方法 -// * @return 执行结果 -// */ -// public Object run(String method) { -// return run(method, new HashMap<>()); -// } -// -// /** -// * 执行cdp方法 -// * -// * @param method 方法 -// * @param params 参数 -// * @return 执行结果 -// */ -// public Object run(String method, Map params) { -// if (stopped.get()) return Map.of("error", "connection disconnected", "type", "connection_error"); -// params = new HashMap<>(params); -// Object timeout1 = params.remove("_timeout"); -// double timeout = timeout1 != null ? Float.parseFloat(timeout1.toString()) : 30.0; -// -// JSONObject result = this.send(Map.of("method", method, "params", params), timeout); -// if (!result.containsKey("result") && result.containsKey("error")) { -// HashMap map = new HashMap<>(); -// map.put("error", result.getJSONObject("error").get("message")); -// map.put("type", result.getOrDefault("type", "call_method_error")); -// map.put("method", method); -// map.put("args", params); -// map.put("timeout", timeout); -// return JSON.toJSONString(map); -// } else { -// return JSON.toJSONString(result.get("result")); -// } -// } -// -// -// /** -// * 启动连接 -// */ -// private void start() { -// this.stopped.set(false); -// try { -//// Request build = new Request(new HttpUrl("socket","","","",80,new ArrayList<>(),null,null,this.websocketUrl), "GET", Headers.of(), null, new HashMap<>()).newBuilder().url(this.websocketUrl).build(); -//// OkHttpClient okHttpClient = new OkHttpClient(); -// ws = new WebSocketClient(); -// ws.setConnectTimeout(60_000); -// ws.start(); -// webSocketSession = ws.connect(new WebSocketAdapter() { -// @Override -// public void onWebSocketClose(int statusCode, String reason) { -// stop(); -// super.onWebSocketClose(statusCode, reason); -// } -// -// @Override -// public void onWebSocketConnect(Session sess) { -// super.onWebSocketConnect(sess); -// } -// -// @Override -// public void onWebSocketError(Throwable cause) { -// super.onWebSocketError(cause); -// } -// -// @Override -// public boolean isConnected() { -// return super.isConnected(); -// } -// -// @Override -// public void onWebSocketText(String message) { -// webSocketMsg.set(message); -// } -// -// }, URI.create(this.websocketUrl)); -// recvThread.start(); -// handleEventThread.start(); -// } catch (Exception e) { -// e.printStackTrace(); -// stop(); -// } -// } -// -// /** -// * 中断连接 -// */ -// public boolean stop() { -// stop1(); -// while (this.recvThread.isAlive() || this.handleEventThread.isAlive()) { -// try { -// Thread.sleep(100); -// } catch (InterruptedException e) { -// throw new RuntimeException(e); -// } -// } -// return true; -// } -// -// /** -// * 中断连接 -// */ -// private void stop1() { -// if (stopped.get()) return; -// stopped.set(true); -// if (ws != null) { -// try { -// ws.stop(); -// } catch (Exception e) { -// throw new RuntimeException(e); -// } -// ws = null; -// } -// try { -// while (!eventQueue.isEmpty()) { -// Map event = eventQueue.poll(); -// MyRunnable method = eventHandlers.get(event.get("method").toString()); -// if (method != null) { -// method.setMessage(event.get("params")); -// method.run(); -// } -// } -// } catch (Exception ignored) { -// } -// eventHandlers.clear(); -// methodResults.clear(); -// eventQueue.clear(); -// if (occupant != null) occupant.onDisconnect(); -// } -// -// public void setCallback(String event, MyRunnable callback) { -// setCallback(event, callback, false); -// } -// -// /** -// * 绑定cdp event和回调方法 -// * -// * @param event 方法名称 -// * @param callback 绑定到cdp event的回调方法 -// * @param immediate 是否要立即处理的动作 -// */ -// public void setCallback(String event, MyRunnable callback, boolean immediate) { -// Map handler = immediate ? immediateEventHandlers : eventHandlers; -// if (callback != null) handler.put(event, callback); -// else handler.remove(event); -// } -} \ No newline at end of file diff --git a/java/src/main/java/com/ll/DrissonPage/base/Driver_org_webSocket.java b/java/src/main/java/com/ll/DrissonPage/base/Driver_org_webSocket.java deleted file mode 100644 index 68780b2..0000000 --- a/java/src/main/java/com/ll/DrissonPage/base/Driver_org_webSocket.java +++ /dev/null @@ -1,424 +0,0 @@ -package com.ll.DrissonPage.base; - -/** - * 驱动 org.java-websocket - * - * @author 陆 - * @address click - * @original DrissionPage - */ - -public class Driver_org_webSocket { -// /** -// * 标签id -// */ -// @Getter -// private final String id; -// /** -// * 浏览器连接地址 -// */ -// @Getter -// private final String address; -// /** -// * 标签页类型 -// */ -// @Getter -// private final String type; -// private final boolean debug; -// private final String websocketUrl; -// private final AtomicInteger curId; -// /** -// * 会话返回值 -// */ -// private final AtomicReference webSocketMsg = new AtomicReference<>(null); -// private final Thread recvThread; -// private final Thread handleEventThread; -// @Getter -// -// private final AtomicBoolean stopped; -// private final BlockingQueue> eventQueue; -// private final BlockingQueue> immediateEventQueue; -// private final Map eventHandlers; -// private final Map immediateEventHandlers; -// private final Map>> methodResults; -// /** -// * 创建这个驱动的对象 -// */ -// @Getter -// @Setter -// private Occupant occupant; -// private boolean alertFlag; -// /** -// * 会话驱动 -// */ -// private WebSocketClient ws; -// private Thread handleImmediateEventThread; -// -// public Driver(String tabId, String tabType, String address) { -// this(tabId, tabType, address, null); -// } -// -// /** -// * 驱动 -// * -// * @param tabId 标签id -// * @param tabType 标签页类型 -// * @param address 浏览器连接地址 -// */ -// public Driver(String tabId, String tabType, String address, Occupant occupant) { -// this.id = tabId; -// this.address = address; -// this.type = tabType; -// this.occupant = occupant; -// this.debug = true; -// this.alertFlag = false; -// this.websocketUrl = "ws://" + address + "/devtools/" + tabType + "/" + tabId; -// this.curId = new AtomicInteger(0); -// this.ws = null; -// -// -// this.recvThread = new Thread(this::recvLoop); -// this.handleEventThread = new Thread(this::handleEventLoop); -// this.recvThread.setDaemon(true); -// this.handleEventThread.setDaemon(true); -// this.handleImmediateEventThread = null; -// -// this.stopped = new AtomicBoolean(); -// -// this.eventHandlers = new ConcurrentHashMap<>(); -// this.immediateEventHandlers = new ConcurrentHashMap<>(); -// this.methodResults = new ConcurrentHashMap<>(); -// this.eventQueue = new LinkedBlockingQueue<>(); -// this.immediateEventQueue = new LinkedBlockingQueue<>(); -// start(); -// } -// -// /** -// * 发送信息到浏览器,并返回浏览器返回的信息 -// * -// * @param message 发送给浏览器的数据 -// * @param timeout 超时时间,为null表示无时间 -// * @return 浏览器返回的数据 -// */ -// private JSONObject send(Map message, double timeout) { -// message = new HashMap<>(message); -// int wsId = curId.incrementAndGet(); -// message.put("id", wsId); -// String messageJson = JSON.toJSONString(message); -// -// if (this.debug) System.out.println("发->" + messageJson); -// //计算等待时间 -// long endTime = (long) (System.currentTimeMillis() + timeout * 1000L); -// LinkedBlockingQueue> value = new LinkedBlockingQueue<>(); -// methodResults.put(wsId, value); -// try { -// ws.send(messageJson); -// if (timeout == 0) { -// methodResults.remove(wsId); -// return new JSONObject(Map.of("id", wsId, "result", Map.of())); -// } -// } catch (WebsocketNotConnectedException e) { -// e.printStackTrace(); -// methodResults.remove(wsId); -// return new JSONObject(Map.of("error", Map.of("message", "connection disconnected"), "type", "connection_error")); -// } -// int i = 5; -// long endTimes = System.currentTimeMillis() + 1000L; -// while (!stopped.get()) { -// try { -// Map result = methodResults.get(wsId).poll(10_000, TimeUnit.MILLISECONDS); -// if (result == null && System.currentTimeMillis() < endTimes) continue; -// if (result == null && i > 0 && System.currentTimeMillis() > endTimes) { -// i--; -// endTimes = System.currentTimeMillis() + 1000L; -// System.out.println("超时丢包:->" + messageJson); -// ws.send(messageJson); -// continue; -// } -// methodResults.remove(wsId); -// if (result == null) throw new NullPointerException(); -// return new JSONObject(result); -// } catch (InterruptedException | NullPointerException | IllegalArgumentException e) { -//// e.printStackTrace(); -// String string = message.get("method").toString(); -// if (alertFlag && string.startsWith("Input.") || string.startsWith("Runtime.")) { -// return new JSONObject(Map.of("error", Map.of("message", "alert exists."), "type", "alert_exists")); -// } -// if (timeout > 0 && System.currentTimeMillis() > endTime) { -// methodResults.remove(wsId); -// return alertFlag ? new JSONObject(Map.of("error", Map.of("message", "alert exists."), "type", "alert_exists")) : new JSONObject(Map.of("error", Map.of("message", "timeout"), "type", "timeout")); -// } -// } -// } -// -// return new JSONObject(Map.of("error", Map.of("message", "connection disconnected"), "type", "connection_error")); -// } -// -// /** -// * 接收浏览器信息的守护线程方法 -// */ -// private void recvLoop() { -// while (!stopped.get()) { -// JSONObject msg; -// try { -// String andSet = webSocketMsg.getAndSet(null); -// if (andSet != null) { -// msg = JSONObject.parseObject(andSet); -// } else continue; -// } catch (Exception e) { -// if (stop()) return; -// return; -// -// } -// if (this.debug) System.out.println("<-收" + msg); -// -// if (msg.containsKey("method")) { -// if (msg.getString("method").startsWith("Page.javascriptDialog")) { -// alertFlag = msg.getString("method").endsWith("Opening"); -// } -// MyRunnable function = immediateEventHandlers.get(msg.getString("method")); -// if (function != null) { -// this.handleImmediateEvent(function, msg.getOrDefault("params", new HashMap<>())); -// } else { -// try { -// eventQueue.put(msg); -// } catch (InterruptedException e) { -// throw new RuntimeException(e); -// } -// } -// } else { -// int i = 1000; -// Integer integer = msg.getInteger("id"); -// while (i-- > 0 && integer != null && !methodResults.containsKey(integer)) { -// try { -// Thread.sleep(10); -// } catch (InterruptedException e) { -// throw new RuntimeException(e); -// } -// } -// if (methodResults.containsKey(integer)) { -// try { -// methodResults.get(integer).put(msg); -// } catch (InterruptedException e) { -// throw new RuntimeException(e); -// } -// } else if (this.debug) { -// System.out.println("未知错误->" + msg); -// -// } -// } -// -// } -// } -// -// /** -// * 当接收到浏览器信息,执行已绑定的方法 -// */ -// private void handleEventLoop() { -// while (!stopped.get()) { -// Map event; -// try { -// event = eventQueue.poll(1, TimeUnit.SECONDS); -// } catch (InterruptedException e) { -// continue; -// } -// -// if (event != null) { -// MyRunnable function = eventHandlers.get(event.get("method").toString()); -// if (function != null) { -// function.setMessage(event.get("params")); -// function.run(); -// } -// } -// this.eventQueue.poll(); -// -// } -// } -// -// private void handleImmediateEventLoop() { -// while (!stopped.get() && !immediateEventQueue.isEmpty()) { -// Map event; -// try { -// event = immediateEventQueue.poll(1, TimeUnit.SECONDS); -// } catch (InterruptedException e) { -// e.printStackTrace(); -// continue; -// } -// if (event != null) { -// MyRunnable function = immediateEventHandlers.get(event.get("method").toString()); -// if (function != null) { -// function.setMessage(event.get("params")); -// function.run(); -// } -// } -// -// } -// } -// -// /** -// * 处理立即执行的动作 -// * -// * @param function 要运行下方法 -// * @param params 方法参数 -// */ -// private void handleImmediateEvent(MyRunnable function, Object params) { -// Map func = new HashMap<>(); -// func.put("method", function); -// func.put("params", params); -// immediateEventQueue.add(func); -// -// if (handleImmediateEventThread == null || !handleImmediateEventThread.isAlive()) { -// handleImmediateEventThread = new Thread(this::handleImmediateEventLoop); -// handleImmediateEventThread.setDaemon(true); -// handleImmediateEventThread.start(); -// } -// } -// -// /** -// * 执行cdp方法 -// * -// * @param method 方法 -// * @return 执行结果 -// */ -// public Object run(String method) { -// return run(method, new HashMap<>()); -// } -// -// /** -// * 执行cdp方法 -// * -// * @param method 方法 -// * @param params 参数 -// * @return 执行结果 -// */ -// public Object run(String method, Map params) { -// if (stopped.get()) return Map.of("error", "connection disconnected", "type", "connection_error"); -// params = new HashMap<>(params); -// Object timeout1 = params.remove("_timeout"); -// double timeout = timeout1 != null ? Float.parseFloat(timeout1.toString()) : 30.0; -// -// JSONObject result = this.send(Map.of("method", method, "params", params), timeout); -// if (!result.containsKey("result") && result.containsKey("error")) { -// HashMap map = new HashMap<>(); -// map.put("error", result.getJSONObject("error").get("message")); -// map.put("type", result.getOrDefault("type", "call_method_error")); -// map.put("method", method); -// map.put("args", params); -// map.put("timeout", timeout); -// return JSON.toJSONString(map); -// } else { -// return JSON.toJSONString(result.get("result")); -// } -// } -// -// -// /** -// * 启动连接 -// */ -// private void start() { -// this.stopped.set(false); -// try { -// ws = new WebSocketClient(new URI(websocketUrl)) { -// @Override -// public void onOpen(ServerHandshake handshakeData) { -// // 处理 WebSocket 打开事件 -// } -// -// @Override -// public void onMessage(String message) { -// //处理返回数据 -// webSocketMsg.set(message); -// } -// -// -// @Override -// public void onClose(int code, String reason, boolean remote) { -// -// System.out.println("关闭" + reason); -// // 关闭事件处理 -// stop(); -// } -// -// @Override -// public void onError(Exception ex) { -// System.out.println("错误" + ex.getMessage()); -// // 错误事件处理 -//// stop(); -// } -// }; -// ws.setConnectionLostTimeout(60); -// ws.connect(); -// //需要睡0.1秒让其等待 -// while (ws != null && !ws.getReadyState().equals(ReadyState.OPEN)) { -// Thread.sleep(10); -// } -// recvThread.start(); -// handleEventThread.start(); -// } catch (Exception e) { -// e.printStackTrace(); -// stop(); -// } -// } -// -// /** -// * 中断连接 -// */ -// public boolean stop() { -// stop1(); -// while (this.recvThread.isAlive() || this.handleEventThread.isAlive()) { -// try { -// Thread.sleep(100); -// } catch (InterruptedException e) { -// throw new RuntimeException(e); -// } -// } -// return true; -// } -// -// /** -// * 中断连接 -// */ -// private void stop1() { -// if (stopped.get()) { -// return; -// } -// stopped.set(true); -// if (ws != null) { -// ws.close(); -// ws = null; -// } -// -// try { -// while (!eventQueue.isEmpty()) { -// Map