// Copyright (C) 2009 The Android Open Source Project
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.

// CGI environment and execution management portions are:
//
// ========================================================================
// Copyright (c) 2006-2009 Mort Bay Consulting Pty. Ltd.
// ------------------------------------------------------------------------
// All rights reserved. This program and the accompanying materials
// are made available under the terms of the Eclipse Public License v1.0
// and Apache License v2.0 which accompanies this distribution.
// The Eclipse Public License is available at
// http://www.eclipse.org/legal/epl-v10.html
// The Apache License v2.0 is available at
// http://www.opensource.org/licenses/apache2.0.php
// You may elect to redistribute this code under either of these licenses.
// ========================================================================

package com.google.gerrit.httpd.gitweb;

import com.google.gerrit.httpd.GitWebConfig;
import com.google.gerrit.httpd.HtmlDomUtil;
import com.google.gerrit.server.config.SitePaths;
import com.google.gwt.user.server.rpc.RPCServletUtils;
import com.google.inject.Inject;
import com.google.inject.Singleton;

import java.io.File;
import java.io.IOException;

import javax.servlet.ServletOutputStream;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

@SuppressWarnings("serial")
abstract class GitWebCssServlet extends HttpServlet {
  @SuppressWarnings("serial")
  @Singleton
  static class Site extends GitWebCssServlet {
    @Inject
    Site(SitePaths paths, GitWebConfig gwc) throws IOException {
      super(paths.site_css, gwc);
    }
  }

  @SuppressWarnings("serial")
  @Singleton
  static class Default extends GitWebCssServlet {
    @Inject
    Default(GitWebConfig gwc) throws IOException {
      super(gwc.getGitwebCSS(), gwc);
    }
  }

  private static final String ENC = "UTF-8";
  private final byte[] raw_css;
  private final byte[] gz_css;

  GitWebCssServlet(final File src, final GitWebConfig gitWebConfig)
      throws IOException {
    final File dir = src.getParentFile();
    final String name = src.getName();
    final String raw = HtmlDomUtil.readFile(dir, name);
    if (raw != null) {
      raw_css = raw.getBytes(ENC);
      gz_css = HtmlDomUtil.compress(raw_css);
    } else {
      raw_css = null;
      gz_css = null;
    }
  }

  @Override
  protected void doGet(final HttpServletRequest req,
      final HttpServletResponse rsp) throws IOException {
    if (raw_css != null) {
      rsp.setContentType("text/css");
      rsp.setCharacterEncoding(ENC);
      final byte[] toSend;
      if (RPCServletUtils.acceptsGzipEncoding(req)) {
        rsp.setHeader("Content-Encoding", "gzip");
        toSend = gz_css;
      } else {
        toSend = raw_css;
      }
      rsp.setContentLength(toSend.length);

      final ServletOutputStream os = rsp.getOutputStream();
      try {
        os.write(toSend);
      } finally {
        os.close();
      }
    } else {
      rsp.sendError(HttpServletResponse.SC_NOT_FOUND);
    }
  }
}
