<html><body style="word-wrap: break-word; -webkit-nbsp-mode: space; -webkit-line-break: after-white-space; "><br><div><div>On 17.02.2010, at 10:53, Maarten Bosteels wrote:</div><br class="Apple-interchange-newline"><blockquote type="cite">Hi Joern,<br><br>The reason for using generics in the Encoder interface is to avoid having to cast. (hmm, I guess that's always the reason to use generics ?)<br><br>Take for example, the ObjectEncoder: it needs an ObjectOutputStream, so it wraps the given OutputStream in an ObjectOutputStream:<br> <br>public class ObjectEncoder implements Encoder<ObjectOutputStream> {<br><br> public void encode(ILoggingEvent event, ObjectOutputStream output) throws IOException {<br> output.writeObject(event);<br> }<br><br> public ObjectOutputStream decorate(OutputStream os) throws IOException {<br> return new ObjectOutputStream(os);<br> }<br>}<br><br><br>On the other hand, the ProtobufEncoder can use a plain OutputStream, so its decorate impl is simply a no-op:<br> <br>public class ProtobufEncoder implements Encoder<OutputStream>{<br><br> private ProtoBufConvertor convertor = new ProtoBufConvertor();<br><br> public void encode(ILoggingEvent event, OutputStream output) throws IOException {<br> LoggingProtos.LoggingEvent ev = convertor.convert(event);<br> ev.writeTo(output);<br> }<br><br> public OutputStream decorate(OutputStream os) throws IOException {<br> return os;<br> }<br>}<br><br>Clear now ?<br></blockquote><div><br></div><div>Yes, understood.</div><div><br></div><div>But if I take a look at</div><div><a href="http://code.google.com/p/firewood/source/browse/trunk/compare-formats/src/test/java/com/googlecode/firewood/encoder/SocketAppender.java">http://code.google.com/p/firewood/source/browse/trunk/compare-formats/src/test/java/com/googlecode/firewood/encoder/SocketAppender.java</a></div><div><br></div><div>...I'm not sure why SocketAppender should be concerned about the implementation of the encoder, or, more precisely, about the stream type that is required by the encoder implementation.</div><div><br></div><div>I also don't know if this would be of any help during appender initialization via Joran since the generic type will be erased.</div><div><br></div><div>If I understand everything correctly then decorate is called once during startup and the decorated stream is reused.</div><div>This could become problematic since not every encoder can be implemented like this.</div><div><br></div><div>If Java serialization is used and events should be randomly accessible without reading all previous events first, then a new ObjectOutputStream must be created for every event that is written. Otherwise they depend on previously written events.</div><div>The same is the case for GZIPInputStream/OutputStream.</div><div><br></div><div>Cheers,</div><div>Jörn.</div><div><br></div><br><blockquote type="cite"> <br>Maarten<br><pre class="prettyprint lang-java"><span class="pln"></span><span class="pun"><br></span></pre><br><div class="gmail_quote">On Tue, Feb 16, 2010 at 11:18 PM, Joern Huxhorn (JIRA) <span dir="ltr"><<a href="mailto:noreply-jira@qos.ch">noreply-jira@qos.ch</a>></span> wrote:<br> <blockquote class="gmail_quote" style="border-left: 1px solid rgb(204, 204, 204); margin: 0pt 0pt 0pt 0.8ex; padding-left: 1ex;"><br> [ <a href="http://jira.qos.ch/browse/LBCORE-128?page=com.atlassian.jira.plugin.system.issuetabpanels:comment-tabpanel&focusedCommentId=11533#action_11533" target="_blank">http://jira.qos.ch/browse/LBCORE-128?page=com.atlassian.jira.plugin.system.issuetabpanels:comment-tabpanel&focusedCommentId=11533#action_11533</a> ]<br> <div class="im"><br> Joern Huxhorn commented on LBCORE-128:<br> --------------------------------------<br> <br> </div><div class="im">My branch was meant to implement what I need in a manner of least possible impact.<br> I detached the implementation of FileAppender from WriterAppender, extending UnsynchronizedAppenderBase instead.<br> <br> </div><div class="im">writerWrite(String s, boolean flush) (which I didn't rename even though there's no writer anymore but a stream instead) is now using convertToBytes(String s) and writeBytes(byte[] bytes, boolean flush), i.e. the appender writes bytes into a stream instead of writing a String into a Writer. Encoding of the String isn't handled by the Writer anymore. Instead, String.getBytes(charset) is used to transform into bytes. The charset is initialized once during startup.<br> <br> I've also taken a look at Maartens Encoder but I fear I don't really get the reason for either the decorate method or the use of generics in that case.<br> <br> In the meantime, I've also implemented streaming encoder/decoder interfaces for sulky/Lilith.<br> <a href="http://github.com/huxi/sulky/tree/master/sulky-codec/src/main/java/de/huxhorn/sulky/codec/streaming/" target="_blank">http://github.com/huxi/sulky/tree/master/sulky-codec/src/main/java/de/huxhorn/sulky/codec/streaming/</a><br> Those are a supplement of my previous byte[]-based encoders/decoders and allow chaining. I use generics to define the type that is encoded/decoded. The decoration of the stream is meant to be encapsulated in the actual implementation.<br> <br> I realize that this does not allow reusing of the same encapsulated stream, though. This means that encoding using Java serialization is less efficient since duplicate strings, for example, are serialized over and over instead of only once, referencing the already serialized string.<br> It has the advantage that written byte[] are not dependent on previously written ones, on the other hand. This is crucial for random event access, in my case.<br> <br> I don't say Maartens design is bad and mine is better. Neither you nor (especially!) Maarten should get this impression.<br> <br> If you are actually aiming for a bigger change than my small patch then it might be a good idea to keep the layout concept, even for binary files, and add the ability to use binary layouts that would also contain the encoder for the event.<br> I've enabled binary headers and footers by declaring writeHeader() & writeFooter() protected instead of private in FileAppender - so I could (re)implement them as needed in my extending class.<br> <br> If Layout would/could handle binary data for header, footer and actual events instead, extending RollingFileAppender wouldn't be necessary anymore. One could simply declare a LilithLayout (with parameters?) instead.<br> This would be a much cleaner design but I feared that such a fundamental change wouldn't have any chance of being accepted for inclusion.<br> <br> Thanks for taking a look at this issue. I really appreciate it!<br> <br> </div><div><div></div><div class="h5">> Please support implementation of binary log files in RollingFileAppender/FileAppender<br> > -------------------------------------------------------------------------------------<br> ><br> > Key: LBCORE-128<br> > URL: <a href="http://jira.qos.ch/browse/LBCORE-128" target="_blank">http://jira.qos.ch/browse/LBCORE-128</a><br> > Project: logback-core<br> > Issue Type: Improvement<br> > Components: Appender<br> > Affects Versions: 0.9.17<br> > Reporter: Joern Huxhorn<br> > Assignee: Ceki Gulcu<br> ><br> > This was discussed briefly at <a href="http://marc.info/?l=logback-dev&m=124905434331308&w=2" target="_blank">http://marc.info/?l=logback-dev&m=124905434331308&w=2</a> and I forgot to file a ticket about this.<br> > Currently, RandomFileAppender => FileAppender => WriterAppender is using the following method in WriterAppender to actually write the data:<br> > protected void writerWrite(String s, boolean flush) throws IOException<br> > Please add an additional method like<br> > protected void writerWrite(byte[] bytes, boolean flush) throws IOException<br> > to write to the underlying stream directly.<br> > writerWrite(String, boolean) could call that method after performing the transformation internally, making this change transparent for the rest of the implementation.<br> > Using a binary format for logfiles could have tremendous performance impact as can be seen here: <a href="http://sourceforge.net/apps/trac/lilith/wiki/SerializationPerformance" target="_blank">http://sourceforge.net/apps/trac/lilith/wiki/SerializationPerformance</a><br> <br> --<br> This message is automatically generated by JIRA.<br> -<br> If you think it was sent incorrectly contact one of the administrators: <a href="http://jira.qos.ch/secure/Administrators.jspa" target="_blank">http://jira.qos.ch/secure/Administrators.jspa</a><br> -<br> For more information on JIRA, see: <a href="http://www.atlassian.com/software/jira" target="_blank">http://www.atlassian.com/software/jira</a><br> <br> <br> _______________________________________________<br> logback-dev mailing list<br> <a href="mailto:logback-dev@qos.ch">logback-dev@qos.ch</a><br> <a href="http://qos.ch/mailman/listinfo/logback-dev" target="_blank">http://qos.ch/mailman/listinfo/logback-dev</a><br> </div></div></blockquote></div><br> _______________________________________________<br>logback-dev mailing list<br><a href="mailto:logback-dev@qos.ch">logback-dev@qos.ch</a><br>http://qos.ch/mailman/listinfo/logback-dev<br></blockquote></div><br></body></html>