2012-09-27 21:08:26 +02:00
|
|
|
#!/usr/bin/env python
|
|
|
|
# encoding: utf-8
|
|
|
|
# Philipp Bender, 2012
|
|
|
|
# Matt Clarkson, 2012
|
|
|
|
|
2013-09-05 06:48:29 +02:00
|
|
|
import re
|
2012-09-27 21:08:26 +02:00
|
|
|
from waflib.Task import Task
|
|
|
|
from waflib.TaskGen import extension
|
|
|
|
|
|
|
|
"""
|
|
|
|
A simple tool to integrate protocol buffers into your build system.
|
|
|
|
|
2013-09-05 07:10:20 +02:00
|
|
|
Example::
|
|
|
|
|
2012-09-27 21:08:26 +02:00
|
|
|
def configure(conf):
|
|
|
|
conf.load('compiler_cxx cxx protoc')
|
|
|
|
|
|
|
|
def build(bld):
|
|
|
|
bld(
|
|
|
|
features = 'cxx cxxprogram'
|
|
|
|
source = 'main.cpp file1.proto proto/file2.proto',
|
|
|
|
include = '. proto',
|
|
|
|
target = 'executable')
|
|
|
|
|
2013-09-05 07:10:20 +02:00
|
|
|
Notes when using this tool:
|
|
|
|
|
|
|
|
- protoc command line parsing is tricky.
|
|
|
|
|
|
|
|
The generated files can be put in subfolders which depend on
|
|
|
|
the order of the include paths.
|
|
|
|
|
|
|
|
Try to be simple when creating task generators
|
|
|
|
containing protoc stuff.
|
|
|
|
|
2012-09-27 21:08:26 +02:00
|
|
|
"""
|
|
|
|
|
|
|
|
class protoc(Task):
|
2013-09-05 07:10:20 +02:00
|
|
|
# protoc expects the input proto file to be an absolute path.
|
2012-09-27 21:08:26 +02:00
|
|
|
run_str = '${PROTOC} ${PROTOC_FLAGS} ${PROTOC_ST:INCPATHS} ${SRC[0].abspath()}'
|
|
|
|
color = 'BLUE'
|
|
|
|
ext_out = ['.h', 'pb.cc']
|
2013-09-05 06:48:29 +02:00
|
|
|
def scan(self):
|
|
|
|
"""
|
|
|
|
Scan .proto dependencies
|
|
|
|
"""
|
|
|
|
node = self.inputs[0]
|
|
|
|
|
|
|
|
nodes = []
|
|
|
|
names = []
|
|
|
|
seen = []
|
|
|
|
|
|
|
|
if not node: return (nodes, names)
|
|
|
|
|
|
|
|
search_paths = [self.generator.path.find_node(x) for x in self.generator.includes]
|
|
|
|
|
|
|
|
def parse_node(node):
|
|
|
|
if node in seen:
|
|
|
|
return
|
|
|
|
seen.append(node)
|
2015-03-03 12:19:25 +01:00
|
|
|
code = node.read().splitlines()
|
2013-09-05 06:48:29 +02:00
|
|
|
for line in code:
|
|
|
|
m = re.search(r'^import\s+"(.*)";.*(//)?.*', line)
|
|
|
|
if m:
|
|
|
|
dep = m.groups()[0]
|
|
|
|
for incpath in search_paths:
|
|
|
|
found = incpath.find_resource(dep)
|
|
|
|
if found:
|
|
|
|
nodes.append(found)
|
|
|
|
parse_node(found)
|
|
|
|
else:
|
|
|
|
names.append(dep)
|
|
|
|
|
|
|
|
parse_node(node)
|
|
|
|
return (nodes, names)
|
2012-09-27 21:08:26 +02:00
|
|
|
|
|
|
|
@extension('.proto')
|
|
|
|
def process_protoc(self, node):
|
|
|
|
cpp_node = node.change_ext('.pb.cc')
|
|
|
|
hpp_node = node.change_ext('.pb.h')
|
|
|
|
self.create_task('protoc', node, [cpp_node, hpp_node])
|
|
|
|
self.source.append(cpp_node)
|
|
|
|
|
|
|
|
if 'cxx' in self.features and not self.env.PROTOC_FLAGS:
|
2015-02-08 18:51:39 +01:00
|
|
|
#self.env.PROTOC_FLAGS = '--cpp_out=%s' % node.parent.get_bld().abspath() # <- this does not work
|
|
|
|
self.env.PROTOC_FLAGS = '--cpp_out=.'
|
2012-09-27 21:08:26 +02:00
|
|
|
|
|
|
|
use = getattr(self, 'use', '')
|
|
|
|
if not 'PROTOBUF' in use:
|
|
|
|
self.use = self.to_list(use) + ['PROTOBUF']
|
|
|
|
|
|
|
|
def configure(conf):
|
|
|
|
conf.check_cfg(package="protobuf", uselib_store="PROTOBUF", args=['--cflags', '--libs'])
|
|
|
|
conf.find_program('protoc', var='PROTOC')
|
|
|
|
conf.env.PROTOC_ST = '-I%s'
|
|
|
|
|