blob: 16e941faa26d66175bc3f297f9b656dc0ef23f94 [file] [log] [blame]
/*
* Copyright 2011 Google Inc.
*
* 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.
*/
// Author: morlovich@google.com (Maksim Orlovich)
//
// This provides the JsCombineFilter class which combines multiple external JS
// scripts into a single one in order to reduce the amount of fetches that need
// to be done.
#ifndef NET_INSTAWEB_REWRITER_PUBLIC_JS_COMBINE_FILTER_H_
#define NET_INSTAWEB_REWRITER_PUBLIC_JS_COMBINE_FILTER_H_
#include "net/instaweb/rewriter/public/rewrite_context.h"
#include "net/instaweb/rewriter/public/rewrite_driver.h"
#include "net/instaweb/rewriter/public/rewrite_filter.h"
#include "net/instaweb/rewriter/public/rewrite_options.h"
#include "net/instaweb/rewriter/public/script_tag_scanner.h"
#include "pagespeed/kernel/base/basictypes.h"
#include "pagespeed/kernel/base/scoped_ptr.h"
#include "pagespeed/kernel/base/string.h"
#include "pagespeed/kernel/base/string_util.h"
#include "pagespeed/kernel/html/html_element.h"
#include "pagespeed/kernel/html/html_node.h"
#include "pagespeed/kernel/util/url_multipart_encoder.h"
#include "pagespeed/kernel/util/url_segment_encoder.h"
namespace pagespeed { namespace js { struct JsTokenizerPatterns; } }
namespace net_instaweb {
class Statistics;
// Implements combining of multiple external JS files into one via the
// following transformation:
//
// <script src="a.js">
// <stuff>
// <script src="b.js">
//
// gets turned into:
//
// <script src="a.js+b.js">
// <script>eval(mod_pagespeed_${hash("a.js")})</script>
// <stuff>
// <script>eval(mod_pagespeed_${hash("b.js")})</script>
//
// where $hash stands for using the active Hasher and tweaking the result to
// be a valid identifier continuation. Further, the combined source file
// has the code:
// var mod_pagespeed_${hash("a.js")} = "code of a.js as a string literal";
// var mod_pagespeed_${hash("b.js")} = "code of b.js as a string literal";
class JsCombineFilter : public RewriteFilter {
public:
static const char kJsFileCountReduction[]; // statistics variable name
// rewrite_driver is the context owning us, and filter_id is the ID we
// are registered under.
explicit JsCombineFilter(RewriteDriver* rewrite_driver);
virtual ~JsCombineFilter();
// Registers the provided statistics variable names with 'statistics'.
static void InitStats(Statistics* statistics);
virtual const char* id() const {
return RewriteOptions::kJavascriptCombinerId;
}
// Returns true if given JavaScript is likely to be in strict mode.
// This is somewhat conservative towards saying yes, as it doesn't
// take finer points of ; grammar into account.
static bool IsLikelyStrictMode(const pagespeed::js::JsTokenizerPatterns* jstp,
StringPiece input);
protected:
// RewriteFilter overrides --- HTML parsing event handlers.
virtual void StartDocumentImpl();
virtual void StartElementImpl(HtmlElement* element);
virtual void EndElementImpl(HtmlElement* element);
virtual void Characters(HtmlCharactersNode* characters);
virtual void Flush();
virtual void DetermineEnabled(GoogleString* disabled_reason);
virtual void IEDirective(HtmlIEDirectiveNode* directive);
virtual const char* Name() const { return "JsCombine"; }
virtual RewriteContext* MakeRewriteContext();
virtual const UrlSegmentEncoder* encoder() const {
return &encoder_;
}
private:
class JsCombiner;
class Context;
friend class JsCombineFilterTest;
void ConsiderJsForCombination(HtmlElement* element,
HtmlElement::Attribute* src);
// Returns JS variable name where code for given URL should be stored.
static GoogleString VarName(const RewriteDriver* rewrite_driver,
const GoogleString& url);
void NextCombination();
Context* MakeContext();
JsCombiner* combiner() const;
ScriptTagScanner script_scanner_;
int script_depth_; // how many script elements we are inside
// current outermost <script> not with JavaScript we are inside, or NULL
HtmlElement* current_js_script_; // owned by the html parser.
scoped_ptr<Context> context_;
UrlMultipartEncoder encoder_;
DISALLOW_COPY_AND_ASSIGN(JsCombineFilter);
};
} // namespace net_instaweb
#endif // NET_INSTAWEB_REWRITER_PUBLIC_JS_COMBINE_FILTER_H_