/* * Copyright (C) 2014 The Project Lombok Authors. * * Permission is hereby granted, free of charge, to any person obtaining a copy * of this software and associated documentation files (the "Software"), to deal * in the Software without restriction, including without limitation the rights * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell * copies of the Software, and to permit persons to whom the Software is * furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included in * all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN * THE SOFTWARE. */ package lombok.core.debug; import java.util.concurrent.TimeUnit; import java.util.concurrent.atomic.AtomicInteger; import org.eclipse.core.runtime.ILog; import org.eclipse.core.runtime.IStatus; import org.eclipse.core.runtime.Platform; import org.eclipse.core.runtime.Status; import org.osgi.framework.Bundle; public class ProblemReporter { public static void info(String msg, Throwable ex) { init(); try { logger.info(msg, ex); } catch (Throwable t) { logger = new TerminalLogger(); logger.info(msg, ex); } } public static void warning(String msg, Throwable ex) { init(); try { logger.warning(msg, ex); } catch (Throwable t) { logger = new TerminalLogger(); logger.warning(msg, ex); } } public static void error(String msg, Throwable ex) { init(); try { logger.error(msg, ex); } catch (Throwable t) { logger = new TerminalLogger(); logger.error(msg, ex); } } private static void init() { if (logger != null) return; try { logger = new EclipseWorkspaceLogger(); } catch (Throwable t) { logger = new TerminalLogger(); } } private static ErrorLogger logger; private interface ErrorLogger { void info(String message, Throwable ex); void warning(String message, Throwable ex); void error(String message, Throwable ex); } private static class TerminalLogger implements ErrorLogger { @Override public void info(String message, Throwable ex) { System.err.println(message); if (ex != null) ex.printStackTrace(); } @Override public void warning(String message, Throwable ex) { System.err.println(message); if (ex != null) ex.printStackTrace(); } @Override public void error(String message, Throwable ex) { System.err.println(message); if (ex != null) ex.printStackTrace(); } } private static class EclipseWorkspaceLogger implements ErrorLogger { private static final String DEFAULT_BUNDLE_NAME = "org.eclipse.jdt.core"; private static final Bundle bundle; private static final int MAX_LOG = 200; private static final long SQUELCH_TIMEOUT = TimeUnit.HOURS.toMillis(1); private static final AtomicInteger counter = new AtomicInteger(); private static volatile long squelchTimeout = 0L; static { bundle = Platform.getBundle(DEFAULT_BUNDLE_NAME); if (bundle == null) throw new NoClassDefFoundError(); // this means some weird RCP build or possible ecj. At any rate, we can't report this way so act as if this isn't an eclipse. } @Override public void info(String message, Throwable error) { msg(IStatus.INFO, message, error); } @Override public void warning(String message, Throwable error) { msg(IStatus.WARNING, message, error); } @Override public void error(String message, Throwable error) { msg(IStatus.ERROR, message, error); } private void msg(int msgType, String message, Throwable error) { int ct = squelchTimeout != 0L ? 0 : counter.incrementAndGet(); boolean printSquelchWarning = false; if (squelchTimeout != 0L) { long now = System.currentTimeMillis(); if (squelchTimeout > now) return; squelchTimeout = now + SQUELCH_TIMEOUT; printSquelchWarning = true; } else if (ct >= MAX_LOG) { squelchTimeout = System.currentTimeMillis() + SQUELCH_TIMEOUT; printSquelchWarning = true; } ILog log = Platform.getLog(bundle); log.log(new Status(msgType, DEFAULT_BUNDLE_NAME, message, error)); if (printSquelchWarning) { log.log(new Status(IStatus.WARNING, DEFAULT_BUNDLE_NAME, "Lombok has logged too many messages; to avoid memory issues, further lombok logs will be squelched for a while. Restart eclipse to start over.")); } } } }