Compare commits
7 Commits
17ab746445
...
178e6a8eb0
Author | SHA1 | Date |
---|---|---|
|
178e6a8eb0 | |
|
7409bb8fa9 | |
|
fa609157c5 | |
|
367cbd659e | |
|
b746f52f67 | |
|
63440b5ea8 | |
|
231a6c9456 |
13
CHANGELOG.md
13
CHANGELOG.md
|
@ -11,6 +11,19 @@ If this version doesn't do what you want, don't use it ... simple.
|
|||
|
||||
Changes from reusedname's v5.3.2 version.
|
||||
|
||||
25.09.27
|
||||
========================
|
||||
* Added BongaCams URL entry warning
|
||||
* Added model online detection for CGF search, sends name instead of image URL if offline
|
||||
Note: Sending name for FC2 will fail to find matches (Japanese), DC cannot send URL to preview image (wrong mimetype).
|
||||
* Implement PAC (Proxy Auto Configuration)
|
||||
* Fix CheckURL for SC (WinkRU)
|
||||
|
||||
25.09.22
|
||||
========================
|
||||
* Adjust -Xmx parameter, increase Heap size
|
||||
* Change default contact sheet filename
|
||||
|
||||
25.09.15
|
||||
========================
|
||||
* Add missing UserAgent parameter to minimal-browser call for some sites
|
||||
|
|
|
@ -8,7 +8,7 @@
|
|||
<parent>
|
||||
<groupId>ctbrec</groupId>
|
||||
<artifactId>master</artifactId>
|
||||
<version>25.9.15</version>
|
||||
<version>25.9.27</version>
|
||||
<relativePath>../master</relativePath>
|
||||
</parent>
|
||||
|
||||
|
|
|
@ -26,6 +26,12 @@
|
|||
<filtered>true</filtered>
|
||||
<destName>ctbrec-no-splash.sh</destName>
|
||||
</file>
|
||||
<file>
|
||||
<source>${project.basedir}/src/assembly/pac.js</source>
|
||||
<outputDirectory>ctbrec</outputDirectory>
|
||||
<filtered>true</filtered>
|
||||
<destName>pac.js</destName>
|
||||
</file>
|
||||
<file>
|
||||
<source>${project.build.directory}/${project.artifactId}-${project.version}.jar</source>
|
||||
<outputDirectory>ctbrec</outputDirectory>
|
||||
|
|
|
@ -26,6 +26,12 @@
|
|||
<filtered>true</filtered>
|
||||
<destName>ctbrec-no-splash.sh</destName>
|
||||
</file>
|
||||
<file>
|
||||
<source>${project.basedir}/src/assembly/pac.js</source>
|
||||
<outputDirectory>ctbrec</outputDirectory>
|
||||
<filtered>true</filtered>
|
||||
<destName>pac.js</destName>
|
||||
</file>
|
||||
<file>
|
||||
<source>${project.build.directory}/${project.artifactId}-${project.version}.jar
|
||||
</source>
|
||||
|
|
|
@ -0,0 +1,9 @@
|
|||
function FindProxyForURL(url, host) {
|
||||
if (shExpMatch(host, "*stripchat.com")) {
|
||||
return "SOCKS 127.0.0.1:1080";
|
||||
} else if (shExpMatch(host, "*chaturbate.com")) {
|
||||
return "PROXY 127.0.0.1:8080";
|
||||
} else {
|
||||
return "DIRECT";
|
||||
}
|
||||
}
|
|
@ -22,6 +22,12 @@
|
|||
<source>${project.build.directory}/ctbrec-no-splash.exe</source>
|
||||
<outputDirectory>ctbrec</outputDirectory>
|
||||
</file>
|
||||
<file>
|
||||
<source>${project.basedir}/src/assembly/pac.js</source>
|
||||
<outputDirectory>ctbrec</outputDirectory>
|
||||
<filtered>true</filtered>
|
||||
<destName>pac.js</destName>
|
||||
</file>
|
||||
<file>
|
||||
<source>${project.build.directory}/${project.artifactId}-${project.version}.jar</source>
|
||||
<outputDirectory>ctbrec</outputDirectory>
|
||||
|
|
|
@ -2,6 +2,7 @@ package ctbrec.ui.menu;
|
|||
|
||||
|
||||
import lombok.extern.slf4j.Slf4j;
|
||||
import ctbrec.AbstractModel;
|
||||
import ctbrec.Config;
|
||||
import ctbrec.Model;
|
||||
import ctbrec.ModelGroup;
|
||||
|
@ -21,7 +22,9 @@ import javafx.scene.control.SeparatorMenuItem;
|
|||
import javafx.scene.control.TabPane;
|
||||
import javafx.scene.input.Clipboard;
|
||||
import javafx.scene.input.ClipboardContent;
|
||||
import java.io.IOException;
|
||||
import java.net.URLEncoder;
|
||||
import java.util.concurrent.ExecutionException;
|
||||
import java.util.List;
|
||||
import java.util.Optional;
|
||||
import java.util.function.Consumer;
|
||||
|
@ -162,13 +165,22 @@ public class ModelMenuContributor {
|
|||
var openOnCamGirlFinder = new MenuItem("CamGirlFinder");
|
||||
openOnCamGirlFinder.setOnAction(e -> {
|
||||
for (Model model : selectedModels) {
|
||||
String preview = model.getPreview();
|
||||
if (preview != null && !preview.isEmpty()) {
|
||||
String query = URLEncoder.encode(preview, UTF_8);
|
||||
DesktopIntegration.open("https://camgirlfinder.net/search?url=" + query);
|
||||
} else {
|
||||
String query = URLEncoder.encode(model.getName(), UTF_8);
|
||||
DesktopIntegration.open("https://camgirlfinder.net/models?m=" + query + "&p=a&g=a");
|
||||
try {
|
||||
String query;
|
||||
String preview = model.getPreview();
|
||||
if (preview != null && !preview.isEmpty() && model.isOnline(true) && !preview.toLowerCase().contains("dreamcam")) {
|
||||
query = URLEncoder.encode(preview, UTF_8);
|
||||
DesktopIntegration.open("https://camgirlfinder.net/search?url=" + query);
|
||||
continue;
|
||||
}
|
||||
query = URLEncoder.encode(model.getName(), UTF_8);
|
||||
DesktopIntegration.open("https://camgirlfinder.net/models?model=" + query + "&platform=&gender=");
|
||||
} catch (IOException | ExecutionException | InterruptedException ex) {
|
||||
// Handle the exception (e.g., log it or show an error dialog)
|
||||
Dialogs.showError(menu.getParentPopup().getScene(),
|
||||
"Error encoding URL",
|
||||
"Failed to encode the model name or preview URL: " + ex.getMessage(),
|
||||
null);
|
||||
}
|
||||
}
|
||||
});
|
||||
|
|
|
@ -85,6 +85,7 @@ public class SettingsTab extends Tab implements TabSelectionListener {
|
|||
private SimpleStringProperty proxyPort;
|
||||
private SimpleStringProperty proxyUser;
|
||||
private SimpleStringProperty proxyPassword;
|
||||
private SimpleStringProperty pacUrl;
|
||||
private SimpleDirectoryProperty recordingsDir;
|
||||
private SimpleListProperty<DirectoryStructure> directoryStructure;
|
||||
private SimpleListProperty<SplitAfterOption> splitAfter;
|
||||
|
@ -169,11 +170,12 @@ public class SettingsTab extends Tab implements TabSelectionListener {
|
|||
maximumResolutionPlayer = new SimpleIntegerProperty(null, "maximumResolutionPlayer", settings.maximumResolutionPlayer);
|
||||
showPlayerStarting = new SimpleBooleanProperty(null, "showPlayerStarting", settings.showPlayerStarting);
|
||||
singlePlayer = new SimpleBooleanProperty(null, "singlePlayer", settings.singlePlayer);
|
||||
proxyType = new SimpleListProperty<>(null, "proxyType", FXCollections.observableList(List.of(DIRECT, HTTP, SOCKS4, SOCKS5)));
|
||||
proxyType = new SimpleListProperty<>(null, "proxyType", FXCollections.observableList(List.of(DIRECT, HTTP, SOCKS4, SOCKS5, PAC)));
|
||||
proxyHost = new SimpleStringProperty(null, "proxyHost", settings.proxyHost);
|
||||
proxyPort = new SimpleStringProperty(null, "proxyPort", settings.proxyPort);
|
||||
proxyUser = new SimpleStringProperty(null, "proxyUser", settings.proxyUser);
|
||||
proxyPassword = new SimpleStringProperty(null, "proxyPassword", settings.proxyPassword);
|
||||
pacUrl = new SimpleStringProperty(null, "pacUrl", settings.pacUrl);
|
||||
recordingsDir = new SimpleDirectoryProperty(null, "recordingsDir", settings.recordingsDir);
|
||||
directoryStructure = new SimpleListProperty<>(null, "recordingsDirStructure",
|
||||
FXCollections.observableList(List.of(FLAT, ONE_PER_MODEL, ONE_PER_GROUP, ONE_PER_RECORDING)));
|
||||
|
@ -337,7 +339,8 @@ public class SettingsTab extends Tab implements TabSelectionListener {
|
|||
Setting.of("Host", proxyHost).needsRestart(),
|
||||
Setting.of("Port", proxyPort).needsRestart(),
|
||||
Setting.of("Username", proxyUser).needsRestart(),
|
||||
Setting.of("Password", proxyPassword).needsRestart())),
|
||||
Setting.of("Password", proxyPassword).needsRestart(),
|
||||
Setting.of("PAC URL", pacUrl, "URL to your Proxy Auto-Config (PAC) file (e.g. http://example.com/pac.js or file:///G:/path/to/pac.js)").needsRestart())),
|
||||
Category.of("Advanced / Devtools",
|
||||
Group.of("Networking",
|
||||
Setting.of("Playlist request timeout (ms)", playlistRequestTimeout, "Timeout in ms for playlist requests"),
|
||||
|
|
|
@ -406,11 +406,20 @@ public abstract class AbstractRecordedModelsTab extends Tab implements TabSelect
|
|||
}
|
||||
|
||||
protected void addModelByUrl(String url) {
|
||||
if (url.toLowerCase().contains("bonga")) {
|
||||
Dialogs.showError(getTabPane().getScene(),
|
||||
"Do not use URLs for BongaCams",
|
||||
"Use 'BongaCams:<model>' where <model> is obtained from the models page:\nCtrl+u, Ctrl+f, \"username\"",
|
||||
null);
|
||||
return;
|
||||
}
|
||||
|
||||
for (Site site : sites) {
|
||||
var newModel = site.createModelFromUrl(url);
|
||||
if (newModel != null) {
|
||||
if (getMarkModelsForLaterRecording()) {
|
||||
new MarkForLaterRecordingAction(modelInputField, List.of(newModel), true, recorder).execute(m -> Platform.runLater(this::reload));
|
||||
new MarkForLaterRecordingAction(modelInputField, List.of(newModel), true, recorder)
|
||||
.execute(m -> Platform.runLater(this::reload));
|
||||
} else {
|
||||
new StartRecordingAction(modelInputField, List.of(newModel), recorder)
|
||||
.execute()
|
||||
|
|
|
@ -8,7 +8,7 @@
|
|||
<parent>
|
||||
<groupId>ctbrec</groupId>
|
||||
<artifactId>master</artifactId>
|
||||
<version>25.9.15</version>
|
||||
<version>25.9.27</version>
|
||||
<relativePath>../master</relativePath>
|
||||
</parent>
|
||||
|
||||
|
@ -41,6 +41,11 @@
|
|||
<groupId>com.fasterxml.jackson.datatype</groupId>
|
||||
<artifactId>jackson-datatype-jsr310</artifactId>
|
||||
</dependency>
|
||||
<dependency>
|
||||
<groupId>org.bidib.com.github.markusbernhardt</groupId>
|
||||
<artifactId>proxy-vole</artifactId>
|
||||
<version>1.1.6</version>
|
||||
</dependency>
|
||||
<dependency>
|
||||
<groupId>org.mapstruct</groupId>
|
||||
<artifactId>mapstruct</artifactId>
|
||||
|
|
|
@ -32,7 +32,8 @@ public class Settings {
|
|||
DIRECT,
|
||||
HTTP,
|
||||
SOCKS4,
|
||||
SOCKS5
|
||||
SOCKS5,
|
||||
PAC
|
||||
}
|
||||
|
||||
public enum SplitStrategy {
|
||||
|
@ -151,6 +152,7 @@ public class Settings {
|
|||
public String proxyPort;
|
||||
public ProxyType proxyType = ProxyType.DIRECT;
|
||||
public String proxyUser;
|
||||
public String pacUrl;
|
||||
public boolean recentlyWatched = true;
|
||||
public List<String> recordLaterTableColumnOrder = new ArrayList<>();
|
||||
public Map<String, Boolean> recordLaterTableColumnVisibility = new HashMap<>();
|
||||
|
|
|
@ -1,5 +1,7 @@
|
|||
package ctbrec.io;
|
||||
|
||||
import com.github.markusbernhardt.proxy.selector.pac.PacProxySelector;
|
||||
import com.github.markusbernhardt.proxy.selector.pac.UrlPacScriptSource;
|
||||
import com.fasterxml.jackson.core.type.TypeReference;
|
||||
import ctbrec.Config;
|
||||
import ctbrec.LoggingInterceptor;
|
||||
|
@ -20,6 +22,7 @@ import java.io.File;
|
|||
import java.io.IOException;
|
||||
import java.net.Authenticator;
|
||||
import java.net.PasswordAuthentication;
|
||||
import java.net.ProxySelector;
|
||||
import java.nio.file.Files;
|
||||
import java.security.KeyManagementException;
|
||||
import java.security.NoSuchAlgorithmException;
|
||||
|
@ -116,6 +119,18 @@ public abstract class HttpClient {
|
|||
Authenticator.setDefault(new SocksProxyAuth(username, password));
|
||||
}
|
||||
break;
|
||||
case PAC:
|
||||
String pacUrl = config.getSettings().pacUrl;
|
||||
if (pacUrl != null && !pacUrl.isEmpty()) {
|
||||
try {
|
||||
UrlPacScriptSource pacSource = new UrlPacScriptSource(pacUrl);
|
||||
ProxySelector pacSelector = new PacProxySelector(pacSource);
|
||||
ProxySelector.setDefault(pacSelector);
|
||||
} catch (Exception e) {
|
||||
log.warn("Failed to load PAC file: {}", e.getMessage());
|
||||
}
|
||||
}
|
||||
break;
|
||||
case DIRECT:
|
||||
default:
|
||||
System.clearProperty(ProxyConstants.HTTP_PROXY_HOST);
|
||||
|
@ -248,6 +263,10 @@ public abstract class HttpClient {
|
|||
}
|
||||
}
|
||||
|
||||
if (config.getSettings().proxyType == ProxyType.PAC) {
|
||||
builder.proxySelector(ProxySelector.getDefault());
|
||||
}
|
||||
|
||||
ProxyType proxyType = config.getSettings().proxyType;
|
||||
if (proxyType == ProxyType.HTTP) {
|
||||
String username = config.getSettings().proxyUser;
|
||||
|
|
|
@ -45,7 +45,7 @@ public class CreateContactSheet extends AbstractPlaceholderAwarePostProcessor {
|
|||
int cols = Integer.parseInt(getConfig().getOrDefault(COLS, "8"));
|
||||
int rows = Integer.parseInt(getConfig().getOrDefault(ROWS, "7"));
|
||||
String color = getConfig().getOrDefault(BACKGROUND, "0x333333");
|
||||
String filename = getConfig().getOrDefault(FILENAME, "contactsheet.jpg");
|
||||
String filename = getConfig().getOrDefault(FILENAME, "$sanitize(${modelName})_$sanitize(${siteName})_$format(${localDateTime},yyyyMMdd-HHmmss).jpg");
|
||||
int thumbWidth = (int) ((totalWidth - (cols + 1) * padding) / (double)cols);
|
||||
|
||||
Path tempDir = createThumbnails(rec, config);
|
||||
|
|
|
@ -6,7 +6,7 @@ import com.iheartradio.m3u8.data.MasterPlaylist;
|
|||
import com.iheartradio.m3u8.data.Playlist;
|
||||
import com.iheartradio.m3u8.data.PlaylistData;
|
||||
import ctbrec.AbstractModel;
|
||||
// import ctbrec.Config;
|
||||
import ctbrec.Config;
|
||||
import ctbrec.StringUtil;
|
||||
import ctbrec.io.HttpException;
|
||||
import ctbrec.io.json.ObjectMapperFactory;
|
||||
|
@ -383,21 +383,23 @@ public class ChaturbateModel extends AbstractModel {
|
|||
|
||||
@Override
|
||||
public boolean exists() throws IOException {
|
||||
Request req = new Request.Builder() // @formatter:off
|
||||
.url(getUrl())
|
||||
.header(USER_AGENT, site.getHttpClient().getEffectiveUserAgent())
|
||||
.header(ACCEPT_LANGUAGE, Locale.ENGLISH.getLanguage())
|
||||
.build(); // @formatter:on
|
||||
Request req = new Request.Builder()
|
||||
.url(getUrl())
|
||||
.header("Accept", "*/*")
|
||||
.header("Accept-Language", Locale.ENGLISH.getLanguage())
|
||||
.header("User-Agent", Config.getInstance().getSettings().httpUserAgent)
|
||||
.header("Referer", getUrl())
|
||||
.header("Origin", getSite().getBaseUrl())
|
||||
.build();
|
||||
try (Response response = getSite().getHttpClient().execute(req)) {
|
||||
if (!response.isSuccessful() && response.code() == 404) {
|
||||
if (response.code() == 404) {
|
||||
return false;
|
||||
} else {
|
||||
String body = response.body().string();
|
||||
boolean banned = body.contains("This room has been banned");
|
||||
boolean deleted = body.contains("This account has been deleted");
|
||||
boolean redirectedToRoot = Objects.equals("/", response.request().url().encodedPath());
|
||||
return !(banned || deleted || redirectedToRoot);
|
||||
}
|
||||
String body = response.body().string();
|
||||
boolean banned = body.contains("This room has been banned");
|
||||
boolean deleted = body.contains("This account has been deleted");
|
||||
boolean redirectedToRoot = Objects.equals("/", response.request().url().encodedPath());
|
||||
return !banned && !deleted && !redirectedToRoot;
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
@ -426,17 +426,21 @@ public class StripchatModel extends AbstractModel {
|
|||
|
||||
@Override
|
||||
public boolean exists() throws IOException {
|
||||
JSONObject jsonResponse = getModelInfo();
|
||||
if (jsonResponse.optString("error").equals("Not Found")) {
|
||||
log.info("Model not found: {}", getName());
|
||||
return false;
|
||||
}
|
||||
if (jsonResponse.has("user")) {
|
||||
JSONObject user = jsonResponse.getJSONObject("user");
|
||||
if (isBanned(user)) {
|
||||
log.info("Model inactive or deleted: {}", getName());
|
||||
try {
|
||||
JSONObject jsonResponse = getModelInfo();
|
||||
if (jsonResponse.has("user")) {
|
||||
JSONObject user = jsonResponse.getJSONObject("user").getJSONObject("user");
|
||||
if (isBanned(user)) {
|
||||
log.info("Model inactive or deleted: {}", getName());
|
||||
return false;
|
||||
}
|
||||
}
|
||||
} catch (HttpException e) {
|
||||
if (e.getResponseCode() == 404) {
|
||||
log.info("Model not found: {}", getName());
|
||||
return false;
|
||||
}
|
||||
throw e;
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
|
|
@ -11,7 +11,7 @@
|
|||
<groupId>ctbrec</groupId>
|
||||
<artifactId>master</artifactId>
|
||||
<packaging>pom</packaging>
|
||||
<version>25.9.15</version>
|
||||
<version>25.9.27</version>
|
||||
|
||||
<modules>
|
||||
<module>../common</module>
|
||||
|
|
|
@ -8,7 +8,7 @@
|
|||
<parent>
|
||||
<groupId>ctbrec</groupId>
|
||||
<artifactId>master</artifactId>
|
||||
<version>25.9.15</version>
|
||||
<version>25.9.27</version>
|
||||
<relativePath>../master</relativePath>
|
||||
</parent>
|
||||
|
||||
|
|
|
@ -0,0 +1,9 @@
|
|||
function FindProxyForURL(url, host) {
|
||||
if (shExpMatch(host, "*stripchat.com")) {
|
||||
return "SOCKS 127.0.0.1:1080";
|
||||
} else if (shExpMatch(host, "*chaturbate.com")) {
|
||||
return "PROXY 127.0.0.1:8080";
|
||||
} else {
|
||||
return "DIRECT";
|
||||
}
|
||||
}
|
|
@ -30,6 +30,12 @@
|
|||
<source>${project.basedir}/LICENSE.txt</source>
|
||||
<outputDirectory>ctbrec</outputDirectory>
|
||||
</file>
|
||||
<file>
|
||||
<source>${project.basedir}/src/assembly/pac.js</source>
|
||||
<outputDirectory>ctbrec</outputDirectory>
|
||||
<filtered>true</filtered>
|
||||
<destName>pac.js</destName>
|
||||
</file>
|
||||
<file>
|
||||
<source>${project.basedir}/../CHANGELOG.md</source>
|
||||
<outputDirectory>ctbrec</outputDirectory>
|
||||
|
|
Loading…
Reference in New Issue