• Home
  • Features
  • Pricing
  • Docs
  • Announcements
  • Sign In

jiangxincode / ApkToolBoxGUI / #1133

22 Aug 2025 02:26PM UTC coverage: 2.974% (-0.1%) from 3.079%
#1133

push

jiangxincode
merge

236 of 7936 relevant lines covered (2.97%)

0.03 hits per line

Source File
Press 'n' to go to next uncovered line, 'b' for previous

0.0
/src/main/java/edu/jiangxin/apktoolbox/main/MainFrame.java
1
package edu.jiangxin.apktoolbox.main;
2

3
import edu.jiangxin.apktoolbox.Version;
4
import edu.jiangxin.apktoolbox.convert.base.BaseConvertPanel;
5
import edu.jiangxin.apktoolbox.convert.color.ColorConvertPanel;
6
import edu.jiangxin.apktoolbox.convert.color.ColorPickerPanel;
7
import edu.jiangxin.apktoolbox.convert.protobuf.unsupervised.ProtobufConvertPanel;
8
import edu.jiangxin.apktoolbox.convert.relationship.RelationShipConvertPanel;
9
import edu.jiangxin.apktoolbox.convert.time.TimeConvertPanel;
10
import edu.jiangxin.apktoolbox.convert.zh2unicode.Zh2UnicodeConvertPanel;
11
import edu.jiangxin.apktoolbox.android.dumpsys.alarm.DumpsysAlarmPanel;
12
import edu.jiangxin.apktoolbox.file.batchrename.BatchRenamePanel;
13
import edu.jiangxin.apktoolbox.file.EncodeConvertPanel;
14
import edu.jiangxin.apktoolbox.file.OsConvertPanel;
15
import edu.jiangxin.apktoolbox.file.checksum.ChecksumPanel;
16
import edu.jiangxin.apktoolbox.file.password.recovery.RecoveryPanel;
17
import edu.jiangxin.apktoolbox.file.duplicate.DuplicateSearchPanel;
18
import edu.jiangxin.apktoolbox.file.zhconvert.ZhConvertPanel;
19
import edu.jiangxin.apktoolbox.help.*;
20
import edu.jiangxin.apktoolbox.android.i18n.I18nAddPanel;
21
import edu.jiangxin.apktoolbox.android.i18n.I18nFindLongestPanel;
22
import edu.jiangxin.apktoolbox.android.i18n.I18nRemovePanel;
23
import edu.jiangxin.apktoolbox.android.monkey.MonkeyPanel;
24
import edu.jiangxin.apktoolbox.help.settings.SettingsPanel;
25
import edu.jiangxin.apktoolbox.pdf.PdfFinderPanel;
26
import edu.jiangxin.apktoolbox.reverse.*;
27
import edu.jiangxin.apktoolbox.android.screenshot.ScreenShotPanel;
28
import edu.jiangxin.apktoolbox.reverse.ApktoolPanel;
29
import edu.jiangxin.apktoolbox.swing.extend.EasyFrame;
30
import edu.jiangxin.apktoolbox.swing.extend.EasyPanel;
31
import edu.jiangxin.apktoolbox.swing.extend.listener.ChangeMenuListener;
32
import edu.jiangxin.apktoolbox.swing.extend.listener.ChangeMenuToUrlListener;
33
import edu.jiangxin.apktoolbox.swing.extend.listener.IPreChangeMenuCallBack;
34
import edu.jiangxin.apktoolbox.swing.extend.plugin.ChangeMenuPreparePluginController;
35
import edu.jiangxin.apktoolbox.swing.extend.plugin.PluginPanel;
36
import edu.jiangxin.apktoolbox.utils.Utils;
37
import org.apache.commons.configuration2.Configuration;
38
import org.apache.commons.lang3.StringUtils;
39
import org.apache.logging.log4j.LogManager;
40
import org.apache.logging.log4j.Logger;
41

42
import javax.swing.*;
43
import javax.swing.border.EmptyBorder;
44
import java.awt.*;
45
import java.awt.event.*;
46
import java.io.Serial;
47
import java.lang.reflect.InvocationTargetException;
48
import java.text.MessageFormat;
49
import java.util.Locale;
50
import java.util.Objects;
51

52
/**
53
 * @author jiangxin
54
 * @author 2018-08-19
55
 */
56
public class MainFrame extends EasyFrame {
57

58
    @Serial
59
    private static final long serialVersionUID = 1L;
60

61
    private JPanel contentPane;
62
    private JMenuBar menuBar;
63

64
    public static void main(String[] args) {
65
        final Logger logger = LogManager.getLogger(MainFrame.class.getSimpleName());
×
66
        boolean isEnvironmentOk = Utils.checkAndInitEnvironment();
×
67
        if (!isEnvironmentOk) {
×
68
            logger.error("Environment is not ready, exit");
×
69
            return;
×
70
        }
71
        EventQueue.invokeLater(() -> {
×
72
            Configuration conf = Utils.getConfiguration();
×
73
            String lookAndFeelClassName = conf.getString("look.and.feel.class.name");
×
74
            if (StringUtils.isEmpty(lookAndFeelClassName)) {
×
75
                lookAndFeelClassName = "com.formdev.flatlaf.FlatDarculaLaf";
×
76
                conf.setProperty("look.and.feel.class.name", lookAndFeelClassName);
×
77
            }
78
            try {
79
                UIManager.setLookAndFeel(lookAndFeelClassName);
×
80
            } catch (UnsupportedLookAndFeelException | ClassNotFoundException | InstantiationException | IllegalAccessException e) {
×
81
                logger.error("setLookAndFeel failed, use default instead", e);
×
82
            }
×
83

84
            String currentLocaleLanguage = conf.getString("locale.language");
×
85
            if (StringUtils.isEmpty(currentLocaleLanguage)) {
×
86
                currentLocaleLanguage = Locale.ENGLISH.getLanguage();
×
87
                conf.setProperty("locale.language", currentLocaleLanguage);
×
88
            }
89
            Locale.setDefault(new Locale(currentLocaleLanguage));
×
90

91
            MainFrame frame = new MainFrame();
×
92

93
            boolean isAlwaysOnTop = conf.getBoolean("always.on.top", false);
×
94
            conf.setProperty("always.on.top", isAlwaysOnTop);
×
95
            frame.setAlwaysOnTop(isAlwaysOnTop);
×
96

97
            frame.setVisible(true);
×
98
        });
×
99
    }
×
100

101
    public MainFrame() {
102
        super();
×
103
        if (SystemTray.isSupported()) {
×
104
            configSystemTray();
×
105
        }
106

107
        setTitle(MessageFormat.format(bundle.getString("main.title"), Version.VERSION));
×
108
        setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
×
109
        setMenuBar();
×
110
        contentPane = new JPanel();
×
111
        contentPane.setBorder(new EmptyBorder(5, 5, 5, 5));
×
112
        contentPane.setLayout(new BoxLayout(contentPane, BoxLayout.Y_AXIS));
×
113
        contentPane.add(Box.createVerticalGlue());
×
114
        EasyPanel initPanel = new AboutPanel();
×
115
        initPanel.init();
×
116
        initPanel.setBorder(BorderFactory.createTitledBorder(bundle.getString("help.about.title")));
×
117
        contentPane.add(initPanel);
×
118
        contentPane.add(Box.createVerticalGlue());
×
119
        setContentPane(contentPane);
×
120
        refreshSizeAndLocation();
×
121
    }
×
122

123
    private void configSystemTray() {
124
        SystemTray systemTray = SystemTray.getSystemTray();
×
125
        PopupMenu popupMenu = new PopupMenu();
×
126
        final MenuItem show = new MenuItem("Open");
×
127
        final MenuItem exit = new MenuItem("Close");
×
128
        ActionListener actionListener = e -> {
×
129
            if (e.getSource() == show) {
×
130
                setExtendedState(JFrame.NORMAL);
×
131
                setVisible(true);
×
132
            }
133
            if (e.getSource() == exit) {
×
134
                dispatchEvent(new WindowEvent(this, WindowEvent.WINDOW_CLOSING));
×
135
            }
136
        };
×
137
        exit.addActionListener(actionListener);
×
138
        show.addActionListener(actionListener);
×
139
        popupMenu.add(show);
×
140
        popupMenu.add(exit);
×
141
        TrayIcon trayIcon = new TrayIcon(image, "系统托盘", popupMenu);
×
142
        trayIcon.setImageAutoSize(true);
×
143
        try {
144
            systemTray.add(trayIcon);
×
145
        } catch (AWTException e) {
×
146
            logger.error("add icon to tray failed");
×
147
        }
×
148
        trayIcon.addMouseListener(new MouseAdapter() {
×
149
            @Override
150
            public void mouseClicked(MouseEvent e) {
151
                if (e.getClickCount() == 2) {
×
152
                    setExtendedState(JFrame.NORMAL);
×
153
                    setVisible(true);
×
154
                }
155
            }
×
156
        });
157
    }
×
158

159
    private void setMenuBar() {
160
        menuBar = new JMenuBar();
×
161
        setJMenuBar(menuBar);
×
162

163
        createReverseMenu();
×
164

165
        createAndroidMenu();
×
166

167
        createPdfMenu();
×
168

169
        createFileMenu();
×
170

171
        createConvertMenu();
×
172

173
        createHelpMenu();
×
174
    }
×
175

176
    private void createHelpMenu() {
177
        JMenu helpMenu = new JMenu(bundle.getString("help.title"));
×
178
        menuBar.add(helpMenu);
×
179

180
        JMenuItem documentMenuItem = new JMenuItem(bundle.getString("help.document.title"));
×
181
        documentMenuItem.addActionListener(new ChangeMenuToUrlListener(Constant.URL_DOCUMENT));
×
182
        helpMenu.add(documentMenuItem);
×
183

184
        JMenuItem settingsMenuItem = new JMenuItem(bundle.getString("help.settings.title"));
×
185
        settingsMenuItem.addActionListener(new ChangeMenuToPanelListener(SettingsPanel.class, settingsMenuItem.getText()));
×
186
        helpMenu.add(settingsMenuItem);
×
187

188
        JMenuItem feedbackMenuItem = new JMenuItem(bundle.getString("help.feedback.title"));
×
189
        feedbackMenuItem.addActionListener(new ChangeMenuToUrlListener(Constant.URL_FEEDBACK));
×
190
        helpMenu.add(feedbackMenuItem);
×
191

192
        JMenuItem checkUpdateMenuItem = new JMenuItem(bundle.getString("help.checkupdate.title"));
×
193
        checkUpdateMenuItem.addActionListener(new CheckUpdateActionListener(this));
×
194
        helpMenu.add(checkUpdateMenuItem);
×
195

196
        JMenuItem contributeMenuItem = new JMenuItem(bundle.getString("help.contribute.title"));
×
197
        contributeMenuItem.addActionListener(new ChangeMenuToUrlListener(Constant.URL_CONTRIBUTE));
×
198
        helpMenu.add(contributeMenuItem);
×
199

200
        JMenuItem aboutMenuItem = new JMenuItem(bundle.getString("help.about.title"));
×
201
        aboutMenuItem.addActionListener(new ChangeMenuToPanelListener(AboutPanel.class, aboutMenuItem.getText()));
×
202
        helpMenu.add(aboutMenuItem);
×
203
    }
×
204

205
    private void createPdfMenu() {
206
        JMenu pdfMenu = new JMenu(bundle.getString("pdf.title"));
×
207
        menuBar.add(pdfMenu);
×
208

209
        JMenuItem pdfFinderMenuItem = new JMenuItem(bundle.getString("pdf.finder.title"));
×
210
        pdfFinderMenuItem.addActionListener(new ChangeMenuToPanelListener(PdfFinderPanel.class, pdfFinderMenuItem.getText()));
×
211
        pdfMenu.add(pdfFinderMenuItem);
×
212
    }
×
213

214
    private void createFileMenu() {
215
        JMenu fileMenu = new JMenu(bundle.getString("file.title"));
×
216
        menuBar.add(fileMenu);
×
217

218
        JMenuItem osConvertMenuItem = new JMenuItem(bundle.getString("file.os.convert.title"));
×
219
        osConvertMenuItem.addActionListener(new ChangeMenuToPanelListener(OsConvertPanel.class, osConvertMenuItem.getText()));
×
220
        fileMenu.add(osConvertMenuItem);
×
221

222
        JMenuItem encodeConvertMenuItem = new JMenuItem(bundle.getString("file.encode.convert.title"));
×
223
        encodeConvertMenuItem.addActionListener(new ChangeMenuToPanelListener(EncodeConvertPanel.class, encodeConvertMenuItem.getText()));
×
224
        fileMenu.add(encodeConvertMenuItem);
×
225

226
        JMenuItem zhConvertMenuItem = new JMenuItem(bundle.getString("file.zh.convert.title"));
×
227
        zhConvertMenuItem.addActionListener(new ChangeMenuToPanelListener(ZhConvertPanel.class, zhConvertMenuItem.getText()));
×
228
        fileMenu.add(zhConvertMenuItem);
×
229

230
        JMenuItem duplicateFindMenuItem = new JMenuItem(bundle.getString("file.duplicate.title"));
×
231
        duplicateFindMenuItem.addActionListener(new ChangeMenuToPanelListener(DuplicateSearchPanel.class, duplicateFindMenuItem.getText()));
×
232
        fileMenu.add(duplicateFindMenuItem);
×
233

234
        JMenuItem batchRenameMenuItem = new JMenuItem(bundle.getString("file.batch.rename.title"));
×
235
        batchRenameMenuItem.addActionListener(new ChangeMenuToPanelListener(BatchRenamePanel.class, batchRenameMenuItem.getText()));
×
236
        fileMenu.add(batchRenameMenuItem);
×
237

238
        JMenuItem checkSumMenuItem = new JMenuItem(bundle.getString("file.check.summary.title"));
×
239
        checkSumMenuItem.addActionListener(new ChangeMenuToPanelListener(ChecksumPanel.class, checkSumMenuItem.getText()));
×
240
        fileMenu.add(checkSumMenuItem);
×
241

242
        JMenuItem recoveryMenuItem = new JMenuItem(bundle.getString("file.password.recovery.title"));
×
243
        recoveryMenuItem.addActionListener(new ChangeMenuToPanelListener(RecoveryPanel.class, recoveryMenuItem.getText()));
×
244
        fileMenu.add(recoveryMenuItem);
×
245
    }
×
246

247
    private void createConvertMenu() {
248
        JMenu convertMenu = new JMenu(bundle.getString("convert.title"));
×
249
        menuBar.add(convertMenu);
×
250

251
        JMenuItem timeConvertMenuItem = new JMenuItem(bundle.getString("convert.time.title"));
×
252
        timeConvertMenuItem.addActionListener(new ChangeMenuToPanelListener(TimeConvertPanel.class, timeConvertMenuItem.getText()));
×
253
        convertMenu.add(timeConvertMenuItem);
×
254

255
        JMenuItem colorConvertMenuItem = new JMenuItem(bundle.getString("convert.color.title"));
×
256
        colorConvertMenuItem.addActionListener(new ChangeMenuToPanelListener(ColorConvertPanel.class, colorConvertMenuItem.getText()));
×
257
        convertMenu.add(colorConvertMenuItem);
×
258

259
        JMenuItem colorPickerMenuItem = new JMenuItem(bundle.getString("picker.color.title"));
×
260
        colorPickerMenuItem.addActionListener(new ChangeMenuToPanelListener(ColorPickerPanel.class, colorPickerMenuItem.getText()));
×
261
        convertMenu.add(colorPickerMenuItem);
×
262

263
        JMenuItem baseConvertMenuItem = new JMenuItem(bundle.getString("convert.base.title"));
×
264
        baseConvertMenuItem.addActionListener(new ChangeMenuToPanelListener(BaseConvertPanel.class, baseConvertMenuItem.getText()));
×
265
        convertMenu.add(baseConvertMenuItem);
×
266

267
        JMenuItem unicodeConvertMenuItem = new JMenuItem(bundle.getString("convert.unicode.title"));
×
268
        unicodeConvertMenuItem.addActionListener(new ChangeMenuToPanelListener(Zh2UnicodeConvertPanel.class, unicodeConvertMenuItem.getText()));
×
269
        convertMenu.add(unicodeConvertMenuItem);
×
270

271
        JMenuItem relationShipConvertMenuItem = new JMenuItem(bundle.getString("convert.relationship.title"));
×
272
        relationShipConvertMenuItem.addActionListener(new ChangeMenuToPanelListener(RelationShipConvertPanel.class, relationShipConvertMenuItem.getText()));
×
273
        convertMenu.add(relationShipConvertMenuItem);
×
274

275
        JMenuItem protobufConvertMenuItem = new JMenuItem(bundle.getString("convert.protobuf.title"));
×
276
        protobufConvertMenuItem.addActionListener(new ChangeMenuToPanelListener(ProtobufConvertPanel.class, protobufConvertMenuItem.getText()));
×
277
        convertMenu.add(protobufConvertMenuItem);
×
278
    }
×
279

280
    private void createAndroidMenu() {
281
        JMenu androidMenu = new JMenu(bundle.getString("android.title"));
×
282
        menuBar.add(androidMenu);
×
283

284
        JMenuItem i18nAddMenuItem = new JMenuItem(bundle.getString("android.i18n.add.title"));
×
285
        i18nAddMenuItem.addActionListener(new ChangeMenuToPanelListener(I18nAddPanel.class, i18nAddMenuItem.getText()));
×
286
        androidMenu.add(i18nAddMenuItem);
×
287

288
        JMenuItem i18nFindLongestMenuItem = new JMenuItem(bundle.getString("android.i18n.longest.title"));
×
289
        i18nFindLongestMenuItem.addActionListener(new ChangeMenuToPanelListener(I18nFindLongestPanel.class, i18nFindLongestMenuItem.getText()));
×
290
        androidMenu.add(i18nFindLongestMenuItem);
×
291

292
        JMenuItem i18nRemoveMenuItem = new JMenuItem(bundle.getString("android.i18n.remove.title"));
×
293
        i18nRemoveMenuItem.addActionListener(new ChangeMenuToPanelListener(I18nRemovePanel.class, i18nRemoveMenuItem.getText()));
×
294
        androidMenu.add(i18nRemoveMenuItem);
×
295

296
        JMenuItem screenShotMenuItem = new JMenuItem(bundle.getString("android.screenshot.title"));
×
297
        screenShotMenuItem.addActionListener(new ChangeMenuToPanelListener(ScreenShotPanel.class, screenShotMenuItem.getText()));
×
298
        screenShotMenuItem.setAccelerator(KeyStroke.getKeyStroke(KeyEvent.VK_S, InputEvent.CTRL_DOWN_MASK));
×
299
        androidMenu.add(screenShotMenuItem);
×
300

301
        JMenuItem monkeyMenuItem = new JMenuItem(bundle.getString("android.monkey.title"));
×
302
        monkeyMenuItem.addActionListener(new ChangeMenuToPanelListener(MonkeyPanel.class, monkeyMenuItem.getText()));
×
303
        androidMenu.add(monkeyMenuItem);
×
304

305
        JMenuItem dumpsysAlarmMenuItem = new JMenuItem(bundle.getString("android.dumpsys.alarm.title"));
×
306
        dumpsysAlarmMenuItem.addActionListener(new ChangeMenuToPanelListener(DumpsysAlarmPanel.class, dumpsysAlarmMenuItem.getText()));
×
307
        androidMenu.add(dumpsysAlarmMenuItem);
×
308
    }
×
309

310
    private void createReverseMenu() {
311
        JMenu reverseMenu = new JMenu(bundle.getString("reverse.title"));
×
312
        reverseMenu.setMnemonic(KeyEvent.VK_R);
×
313
        menuBar.add(reverseMenu);
×
314

315
        JMenuItem pluginVersionMenuItem = new JMenuItem(bundle.getString("reverse.plugin.version.title"));
×
316
        pluginVersionMenuItem.addActionListener(new ChangeMenuToUrlListener(Constant.URL_PLUGIN_VERSION));
×
317
        reverseMenu.add(pluginVersionMenuItem);
×
318

319
        JMenuItem apktoolMenuItem = new JMenuItem(bundle.getString("reverse.apktool.title"), KeyEvent.VK_D);
×
320
        apktoolMenuItem.addActionListener(new ChangeMenuToPanelListener(ApktoolPanel.class, apktoolMenuItem.getText()));
×
321
        reverseMenu.add(apktoolMenuItem);
×
322

323
        JMenuItem apkSignMenuItem = new JMenuItem(bundle.getString("reverse.apksigner.title"));
×
324
        apkSignMenuItem.addActionListener(new ChangeMenuToPanelListener(ApkSignerPanel.class, apkSignMenuItem.getText()));
×
325
        reverseMenu.add(apkSignMenuItem);
×
326

327
        JMenuItem jDMenuItem = new JMenuItem(bundle.getString("reverse.jd.gui.title"));
×
328
        jDMenuItem.addActionListener(new ChangeMenToPluginJdListener());
×
329
        reverseMenu.add(jDMenuItem);
×
330

331
        JMenuItem luytenMenuItem = new JMenuItem(bundle.getString("reverse.luyten.title"));
×
332
        luytenMenuItem.addActionListener(new ChangeMenuToPluginLuytenListener());
×
333
        reverseMenu.add(luytenMenuItem);
×
334

335
        JMenuItem jdDuoMenuItem = new JMenuItem(bundle.getString("reverse.jd.duo.title"));
×
336
        jdDuoMenuItem.addActionListener(new ChangeMenuToPluginJdDuoListener());
×
337
        reverseMenu.add(jdDuoMenuItem);
×
338

339
        JMenuItem jdaMenuItem = new JMenuItem(bundle.getString("reverse.jda.title"));
×
340
        jdaMenuItem.addActionListener(new ChangeMenuToPluginJdaListener());
×
341
        reverseMenu.add(jdaMenuItem);
×
342

343
        JMenuItem jADXMenuItem = new JMenuItem(bundle.getString("reverse.jadx.title"));
×
344
        jADXMenuItem.addActionListener(new ChangeMenuToPluginJadxListener());
×
345
        reverseMenu.add(jADXMenuItem);
×
346

347
        JMenuItem aXMLPrinter = new JMenuItem(bundle.getString("reverse.axmlprinter.title"));
×
348
        aXMLPrinter.addActionListener(new ChangeMenuToPanelListener(AxmlPrinterPanel.class, aXMLPrinter.getText()));
×
349
        reverseMenu.add(aXMLPrinter);
×
350
    }
×
351

352
    class ChangeMenuToPanelListener implements ChangeMenuListener {
353

354
        Class<? extends EasyPanel> easyPanelClass;
355

356
        EasyPanel panel = null;
×
357

358
        String title;
359

360
        public ChangeMenuToPanelListener(Class<? extends EasyPanel> easyPanelClass, String title) {
×
361
            this.easyPanelClass = easyPanelClass;
×
362
            this.title = title;
×
363
            panel = createEasyPanel();
×
364
        }
×
365

366
        @Override
367
        public boolean isNeedPreChangeMenu() {
368
            return panel.isNeedPreChangeMenu();
×
369
        }
370

371
        @Override
372
        public void onPreChangeMenu(IPreChangeMenuCallBack callBack) {
373
            if (panel instanceof PluginPanel pluginPanel) {
×
374
                pluginPanel.preparePlugin(new ChangeMenuPreparePluginController(pluginPanel.getPluginFilename(), pluginPanel.isPluginNeedUnzip(), pluginPanel.isPluginNeedUnzipToSeparateDir(), callBack));
×
375
            }
376
        }
×
377

378
        @Override
379
        public void onChangeMenu() {
380
            contentPane.removeAll();
×
381
            contentPane.add(Box.createVerticalGlue());
×
382
            panel.init();
×
383
            panel.setBorder(BorderFactory.createTitledBorder(title));
×
384
            contentPane.add(panel);
×
385
            logger.info("Panel changed: " + panel.getClass().getSimpleName());
×
386
            contentPane.add(Box.createVerticalGlue());
×
387
            contentPane.revalidate();
×
388
            contentPane.repaint();
×
389
            refreshSizeAndLocation();
×
390
        }
×
391

392
        private EasyPanel createEasyPanel() {
393
            if (easyPanelClass == null) {
×
394
                return new EasyPanel();
×
395
            }
396
            EasyPanel retEasyPanel = null;
×
397
            try {
398
                retEasyPanel = easyPanelClass.getDeclaredConstructor().newInstance();
×
399
            } catch (InstantiationException | IllegalAccessException | InvocationTargetException |
×
400
                     NoSuchMethodException e) {
401
                logger.info("createEasyPanel failed: {}", e.getMessage());
×
402
            }
×
403
            return Objects.requireNonNullElseGet(retEasyPanel, EasyPanel::new);
×
404
        }
405
    }
406
}
407

408

STATUS · Troubleshooting · Open an Issue · Sales · Support · CAREERS · ENTERPRISE · START FREE · SCHEDULE DEMO
ANNOUNCEMENTS · TWITTER · TOS & SLA · Supported CI Services · What's a CI service? · Automated Testing

© 2026 Coveralls, Inc